TDME2 1.9.121
Application.cpp
Go to the documentation of this file.
1#if defined(_MSC_VER)
2 // this suppresses a warning redefinition of APIENTRY macro
3 #define NOMINMAX
4 #include <windows.h>
5#endif
6#define GLFW_INCLUDE_NONE
7#include <GLFW/glfw3.h>
8
9#if defined(_WIN32)
10 #define GLFW_EXPOSE_NATIVE_WIN32
11 #include <GLFW/glfw3native.h>
12#endif
13
14#if defined(_WIN32)
15 #include <windows.h>
16 #include <dbghelp.h>
17 #include <stdio.h>
18 #include <string.h>
20#endif
21
22#if defined(__APPLE__)
23 #include <Carbon/Carbon.h>
24#endif
25
26#if !defined(_MSC_VER)
27 #include <dlfcn.h>
28#endif
29
30#include <stdlib.h>
31
32#include <array>
33#include <memory>
34#include <string>
35
36#include <tdme/tdme.h>
37
43#include <tdme/engine/Engine.h>
50#include <tdme/utilities/Hex.h>
51#include <tdme/utilities/RTTI.h>
54#include <tdme/utilities/Time.h>
55
56using std::array;
57using std::shared_ptr;
58using std::string;
59using std::to_string;
60
78
79Renderer* Application::renderer = nullptr;
80Application* Application::application = nullptr;
81InputEventHandler* Application::inputEventHandler = nullptr;
82int64_t Application::timeLast = -1L;
83bool Application::limitFPS = true;
84
85GLFWwindow* Application::glfwWindow = nullptr;
86array<unsigned int, 10> Application::glfwMouseButtonDownFrames;
87int Application::glfwMouseButtonLast = -1;
88bool Application::glfwCapsLockEnabled = false;
89GLFWcursor* Application::glfwHandCursor = nullptr;
90
91string Application::execute(const string& command) {
92 // see: https://stackoverflow.com/questions/478898/how-to-execute-a-command-and-get-output-of-command-within-c-using-posix
93 array<char, 128> buffer;
94 string result;
95 #if defined(_MSC_VER)
96 shared_ptr<FILE> pipe(_popen(command.c_str(), "r"), _pclose);
97 #else
98 shared_ptr<FILE> pipe(popen(command.c_str(), "r"), pclose);
99 #endif
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();
104 }
105 return result;
106}
107
108void Application::executeBackground(const string& command) {
109 #if defined(_WIN32)
110 system(("start " + command).c_str());
111 #else
112 system((command + " </dev/null &>/dev/null &").c_str());
113 #endif
114}
115
116void Application::openBrowser(const string& url) {
117 #if defined(_WIN32)
118 execute("explorer \"" + url + "\"");
119 #elif defined(__APPLE__) || defined(__HAIKU__)
120 execute("open \"" + url + "\"");
121 #else
122 execute("xdg-open \"" + url + "\"");
123 #endif
124}
125
127 glfwSetWindowShouldClose(glfwWindow, GLFW_FALSE);
128}
129
130void Application::exit(int exitCode) {
131 if (Application::application == nullptr) {
133 } else {
136 } else {
138 glfwSetWindowShouldClose(glfwWindow, GLFW_TRUE);
139 }
140 }
141}
142
143#if defined(_WIN32)
144 LONG WINAPI windowsExceptionHandler(struct _EXCEPTION_POINTERS* exceptionInfo) {
145 // see:
146 // https://asmaloney.com/2017/08/code/crash-reporting-for-mingw-32-windows-and-clang-macos-with-qt/
147 // https://www.gamedev.net/forums/topic/478943-stackwalk64-and-x86/
148 // https://stackoverflow.com/questions/12280472/stackwalk64-does-not-work-with-release-build
149
150 static tdme::os::threading::Mutex mutex("windowsexceptionhandler");
151
152 mutex.lock();
153
154 HANDLE process = GetCurrentProcess();
155 HANDLE thread = GetCurrentThread();
156
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);
161
162 CHAR _pathToExecutable[MAX_PATH];
163 GetModuleFileName(GetModuleHandleW(NULL), _pathToExecutable, MAX_PATH);
164 string pathToExecutable = _pathToExecutable;
165
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!");
170 mutex.unlock();
171
172 //
173 Console::println();
174
175 // shutdown console
176 Console::shutdown();
177
178 //
179 return EXCEPTION_EXECUTE_HANDLER;
180 }
181 addr2lineToolCmd = "\"" + StringTools::replace(addr2lineToolCmd, "\\", "\\\\") + "\"";
182 #endif
183
184 auto backtraceExecutableLine = "windowsExceptionHandler(): path to executable: " + pathToExecutable;
185 Console::println(backtraceExecutableLine);
186 backtraceLines.push_back(backtraceExecutableLine);
187
188 pathToExecutable = string("\"") + StringTools::replace(pathToExecutable, "\\", "\\\\") + "\"";
189
190 SymSetOptions(
191 SYMOPT_UNDNAME |
192 SYMOPT_LOAD_LINES |
193 SYMOPT_OMAP_FIND_NEAREST |
194 SYMOPT_DEFERRED_LOADS |
195 SymGetOptions()
196 );
197
198 SymInitialize(
199 process,
200 nullptr,
201 true
202 );
203
204 STACKFRAME64 stackFrame;
205 memset(&stackFrame, 0, sizeof(STACKFRAME64));
206
207 CONTEXT context;
208 memset(&context, 0, sizeof(context));
209 context.ContextFlags = CONTEXT_FULL;
210 RtlCaptureContext(&context);
211
212 DWORD image;
213 #ifdef _M_IX86
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;
221 #elif _M_X64
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;
229 #elif _M_IA64
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;
239 #else
240 #error "Machine not supported"
241 #endif
242
243 // construct backtrace and save it to console.log and crash.log
244 int frameIdx = 1;
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;
248
249 auto line = 0;
250 char _fileName[1024];
251 char _functionName[1024];
252
253 // Get function name
254 {
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, "??");
264 } else {
265 pSymbol->Name[cnBufferSize-1] = '\0';
266 strcpy(_functionName, pSymbol->Name);
267 }
268 }
269
270 // Get file/line number
271 {
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))
277 {
278 strcpy(_fileName, "??");
279 }
280 else
281 {
282 const char* pszFile = strrchr(theLine.FileName, '\\');
283 if (!pszFile) pszFile = theLine.FileName; else ++pszFile;
284 strncpy(_fileName, pszFile, cnBufferSize);
285 line = theLine.LineNumber;
286 }
287 }
288
289 //
290 string functionName = _functionName;
291 string fileName = _fileName;
292 #if defined(_MSC_VER) == false
293 if (functionName == "??") {
294 string hexAddr;
295 Hex::encodeInt(stackFrame.AddrPC.Offset, hexAddr);
296 string addr2LineCommand = "\"" + addr2lineToolCmd + " -f -p -e " + string(pathToExecutable) + " " + hexAddr + "\"";
297 auto addr2LineOutput = Application::execute(addr2LineCommand);
299 t.tokenize(addr2LineOutput, " ");
300 if (t.hasMoreTokens() == true) {
301 auto addr2lineFunctionName = t.nextToken();
302 addr2LineOutput = StringTools::replace(StringTools::replace(addr2LineOutput, addr2lineFunctionName, RTTI::demangle(addr2lineFunctionName.c_str())), "\n", "");
303 }
304 auto backtraceLine = to_string(frameIdx) + ": " + addr2LineOutput;
305 Console::println(backtraceLine);
306 backtraceLines.push_back(backtraceLine);
307 } else {
308 auto backtraceLine = to_string(frameIdx) + ": " + string(RTTI::demangle(_functionName)) + " at " + fileName + ":" + to_string(line);
309 Console::println(backtraceLine);
310 backtraceLines.push_back(backtraceLine);
311 }
312 #else
313 Console::println(to_string(frameIdx) + ": " + functionName + " at " + fileName + ":" + to_string(line));
314 #endif
315 frameIdx++;
316 }
317
318 // store also to crash.log
319 FileSystem::getInstance()->setContentFromStringArray(".", "crash.log", backtraceLines);
320
321 SymCleanup(process);
322
323 mutex.unlock();
324
325 Console::println();
326
327 // shutdown console
328 Console::shutdown();
329
330 //
331 return EXCEPTION_EXECUTE_HANDLER;
332 }
333#endif
334
338}
339
341}
342
344 Engine::renderer->setVSync(vSync);
345 if (Engine::renderer->getRendererType() != Renderer::RENDERERTYPE_VULKAN) {
346 glfwSwapInterval(vSync == true?1:0);
347 }
348}
349
351 #if defined(__FreeBSD__)
352 return "FreeBSD";
353 #elif defined(__HAIKU__)
354 return "Haiku";
355 #elif defined(__linux__)
356 return "Linux";
357 #elif defined(__APPLE__)
358 return "MacOSX";
359 #elif defined(__NetBSD__)
360 return "NetBSD";
361 #elif defined(__OpenBSD__)
362 return "OpenBSD";
363 #elif defined(_MSC_VER)
364 return "Windows-MSC";
365 #elif defined(_WIN32)
366 return "Windows-MINGW";
367 #else
368 return "Unknown";
369 #endif
370}
371
373 #if defined(__amd64__) || defined(_M_X64)
374 return "x64";
375 #elif defined(__ia64__) || defined(_M_IA64)
376 return "ia64";
377 #elif defined(__aarch64__)
378 return "arm64";
379 #elif defined(__arm__) || defined(_M_ARM)
380 return "arm";
381 #elif defined(__powerpc64__)
382 return "ppc64";
383 #elif defined(__powerpc__)
384 return "ppc";
385 #else
386 return "Unknown";
387 #endif
388}
389
392}
393
395 #if defined(_WIN32)
396 return GetActiveWindow() == GetForegroundWindow();
397 #else
398 return true;
399 #endif
400}
401
403 return windowXPosition;
404}
405
406void Application::setWindowXPosition(int windowXPosition) {
407 this->windowXPosition = windowXPosition;
408}
409
411 return windowYPosition;
412}
413
414void Application::setWindowYPosition(int windowYPosition) {
415 this->windowYPosition = windowYPosition;
416}
417
419 return windowWidth;
420}
421
422void Application::setWindowWidth(int windowWidth) {
423 this->windowWidth = windowWidth;
424 if (initialized == true) {
425 if (fullScreen == false) glfwSetWindowSize(glfwWindow, windowWidth, windowHeight);
426 }
427}
428
430 return windowHeight;
431}
432
433void Application::setWindowHeight(int windowHeight) {
434 this->windowHeight = windowHeight;
435 if (initialized == true) {
436 if (fullScreen == false) glfwSetWindowSize(glfwWindow, windowWidth, windowHeight);
437 }
438}
439
441 return fullScreen;
442}
443
444void Application::setFullScreen(bool fullScreen) {
445 this->fullScreen = fullScreen;
446 if (initialized == true) {
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);
452 } else
453 if (windowMonitor != nullptr && fullScreen == false) {
454 glfwSetWindowMonitor(glfwWindow, NULL, windowXPosition, windowYPosition, windowWidth, windowHeight, 0);
456 glfwSetWindowPos(glfwWindow, windowXPosition, windowYPosition);
457 glfwGetWindowSize(glfwWindow, &windowWidth, &windowHeight);
459 }
460 }
461 }
462}
463
465 #if defined(_WIN32)
466 SetUnhandledExceptionFilter(windowsExceptionHandler);
467 #endif
468}
469
470void Application::setMouseCursor(int mouseCursor) {
471 if (mouseCursor == MOUSE_CURSOR_DISABLED) {
472 glfwSetCursor(glfwWindow, nullptr);
473 glfwSetInputMode(glfwWindow, GLFW_CURSOR, GLFW_CURSOR_HIDDEN);
474 } else
475 if (mouseCursor == MOUSE_CURSOR_ENABLED) {
476 glfwSetCursor(glfwWindow, nullptr);
477 glfwSetInputMode(glfwWindow, GLFW_CURSOR, GLFW_CURSOR_NORMAL);
478 } else
479 if (mouseCursor == MOUSE_CURSOR_HAND) {
480 glfwSetCursor(glfwWindow, glfwHandCursor);
481 glfwSetInputMode(glfwWindow, GLFW_CURSOR, GLFW_CURSOR_NORMAL);
482 }
483}
484
486 double mouseX, mouseY;
487 glfwGetCursorPos(glfwWindow, &mouseX, &mouseY);
488 return static_cast<int>(mouseX);
489}
490
492 double mouseX, mouseY;
493 glfwGetCursorPos(glfwWindow, &mouseX, &mouseY);
494 return static_cast<int>(mouseY);
495}
496
498 #if defined(__APPLE__)
499 int windowXPos, windowYPos;
500 glfwGetWindowPos(Application::glfwWindow, &windowXPos, &windowYPos);
501 CGPoint point;
502 point.x = windowXPos + x;
503 point.y = windowYPos + y;
504 CGWarpMouseCursorPosition(point);
505 CGAssociateMouseAndMouseCursorPosition(true);
506 #else
507 glfwSetCursorPos(glfwWindow, x, y);
508 #endif
509}
510
512 if (Engine::renderer->getRendererType() != Renderer::RENDERERTYPE_VULKAN) glfwSwapBuffers(glfwWindow);
513}
514
516 return string(glfwGetClipboardString(glfwWindow));
517}
518
519void Application::setClipboardContent(const string& content) {
520 glfwSetClipboardString(glfwWindow, content.c_str());
521}
522
523static void glfwErrorCallback(int error, const char* description) {
524 Console::println(string("glfwErrorCallback(): ") + description);
525}
526
527void Application::run(int argc, char** argv, const string& title, InputEventHandler* inputEventHandler, int windowHints) {
528 string rendererLibrary = "libopengl3corerenderer";
529 for (auto i = 1; i < argc; i++) {
530 auto argValue = string(argv[i]);
531 if (argValue == "--debug") debuggingEnabled = true; else
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";
536 }
537
538 #if defined(_WIN32)
539 rendererLibrary = rendererLibrary + ".dll";
540 #elif defined(__APPLE__)
541 rendererLibrary = rendererLibrary + ".dylib";
542 #else
543 rendererLibrary = rendererLibrary + ".so";
544 #endif
545
546 //
547 this->title = title;
548 this->windowHints = windowHints;
549 executableFileName = FileSystem::getInstance()->getFileName(argv[0]);
551 glfwSetErrorCallback(glfwErrorCallback);
552 if (glfwInit() == false) {
553 Console::println("glflInit(): failed!");
554 return;
555 }
556
557 // TODO: dispose
558 Application::glfwHandCursor = glfwCreateStandardCursor(GLFW_HAND_CURSOR);
559
560 // determine window position of not yet done
561 if (windowXPosition == -1 || windowYPosition == -1) {
562 // have some random position if position determination does fail
563 windowXPosition = 100;
564 windowYPosition = 100;
565 // otherwise center application window on primary monitor
566 GLFWmonitor* monitor = glfwGetPrimaryMonitor();
567 const GLFWvidmode* monitorMode = glfwGetVideoMode(monitor);
568 if (monitorMode != nullptr) {
569 int monitorX = -1;
570 int monitorY = -1;
571 glfwGetMonitorPos(monitor, &monitorX, &monitorY);
572 windowXPosition = monitorX + (monitorMode->width - windowWidth) / 2;
573 windowYPosition = monitorY + (monitorMode->height - windowHeight) / 2;
574 }
575 }
576
577 Console::println("Application::run(): Opening renderer library: " + rendererLibrary);
578
579 // load renderer library
580 #if defined(_MSC_VER)
581 //
582 auto rendererLibraryHandle = LoadLibrary(rendererLibrary.c_str());
583 if (rendererLibraryHandle == nullptr) {
584 Console::println("Application::run(): Could not open renderer library");
585 glfwTerminate();
586 return;
587 }
588 //
589 Renderer* (*rendererCreateInstance)() = (Renderer*(*)())GetProcAddress(rendererLibraryHandle, "createInstance");
590 //
591 if (rendererCreateInstance == nullptr) {
592 Console::println("Application::run(): Could not find renderer library createInstance() entry point");
593 glfwTerminate();
594 return;
595 }
596 //
597 renderer = (Renderer*)rendererCreateInstance();
598 if (renderer == nullptr) {
599 Console::println("Application::run(): Could not create renderer");
600 glfwTerminate();
601 return;
602 }
603 #else
604 //
605 #if defined(__HAIKU__)
606 auto rendererLibraryHandle = dlopen(("lib/" + rendererLibrary).c_str(), RTLD_NOW);
607 #else
608 auto rendererLibraryHandle = dlopen(rendererLibrary.c_str(), RTLD_NOW);
609 #endif
610 if (rendererLibraryHandle == nullptr) {
611 Console::println("Application::run(): Could not open renderer library");
612 glfwTerminate();
613 return;
614 }
615 //
616 Renderer* (*rendererCreateInstance)() = (Renderer*(*)())dlsym(rendererLibraryHandle, "createInstance");
617 //
618 if (rendererCreateInstance == nullptr) {
619 Console::println("Application::run(): Could not find renderer library createInstance() entry point");
620 glfwTerminate();
621 return;
622 }
623 //
624 renderer = (Renderer*)rendererCreateInstance();
625 if (renderer == nullptr) {
626 Console::println("Application::run(): Could not create renderer");
627 glfwTerminate();
628 return;
629 }
630 #endif
631
632 // window hints
633 if ((windowHints & WINDOW_HINT_NOTRESIZEABLE) == WINDOW_HINT_NOTRESIZEABLE) glfwWindowHint(GLFW_RESIZABLE, GLFW_FALSE);
634 if ((windowHints & WINDOW_HINT_NOTDECORATED) == WINDOW_HINT_NOTDECORATED) glfwWindowHint(GLFW_DECORATED, GLFW_FALSE);
635 if ((windowHints & WINDOW_HINT_INVISIBLE) == WINDOW_HINT_INVISIBLE) glfwWindowHint(GLFW_VISIBLE, GLFW_FALSE);
636 if ((windowHints & WINDOW_HINT_MAXIMIZED) == WINDOW_HINT_MAXIMIZED) glfwWindowHint(GLFW_MAXIMIZED, GLFW_TRUE);
637
638 //
639 for (auto i = 0; renderer->prepareWindowSystemRendererContext(i) == true; i++) {
640 glfwWindow = glfwCreateWindow(windowWidth, windowHeight, title.c_str(), NULL, NULL);
641 if (glfwWindow != nullptr) break;
642 }
643
644 //
645 if (glfwWindow == nullptr) {
646 Console::println("glfwCreateWindow(): Could not create window");
647 glfwTerminate();
648 return;
649 }
650
651 //
653 Console::println("glfwCreateWindow(): Could not initialize window system renderer context");
654 glfwTerminate();
655 return;
656 }
657
658 //
660
661 //
662 setIcon();
663
664 //
665 glfwSetCharCallback(glfwWindow, Application::glfwOnChar);
666 glfwSetKeyCallback(glfwWindow, Application::glfwOnKey);
667 glfwSetCursorPosCallback(glfwWindow, Application::glfwOnMouseMoved);
668 glfwSetMouseButtonCallback(glfwWindow, Application::glfwOnMouseButton);
669 glfwSetScrollCallback(glfwWindow, Application::glfwOnMouseWheel);
670 glfwSetWindowSizeCallback(glfwWindow, Application::glfwOnWindowResize);
671 glfwSetWindowCloseCallback(glfwWindow, Application::glfwOnClose);
673 glfwGetWindowPos(glfwWindow, &windowXPosition, &windowYPosition);
674 glfwGetWindowSize(glfwWindow, &windowWidth, &windowHeight);
676 }
677 #if defined(__APPLE__)
678 // change working directory on MacOSX if started from app bundle
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);
684 }
685 #endif
686 while (glfwWindowShouldClose(glfwWindow) == false) {
688 if (Engine::renderer->getRendererType() != Renderer::RENDERERTYPE_VULKAN) glfwSwapBuffers(glfwWindow);
689 glfwPollEvents();
690 }
691 glfwTerminate();
692 if (Application::application != nullptr) {
693 Console::println("Application::run(): Shutting down application");
696 Application::application = nullptr;
697 }
698 if (exitCode != 0) ::exit(exitCode);
699}
700
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);
717 }
718 GLFWimage glfwIcon;
719 glfwIcon.width = texture->getTextureWidth();
720 glfwIcon.height = texture->getTextureHeight();
721 glfwIcon.pixels = glfwPixels;
722 glfwSetWindowIcon(glfwWindow, 1, &glfwIcon);
723 texture->releaseReference();
724 delete [] glfwPixels;
725 }
726}
727
734 }
735 int64_t timeNow = Time::getCurrentMillis();
736 int64_t timeFrame = 1000/Application::FPS;
737 if (Application::timeLast != -1L) {
738 int64_t timePassed = timeNow - timeLast;
739 if (limitFPS == true && timePassed < timeFrame) Thread::sleep(timeFrame - timePassed);
740 }
741 Application::timeLast = timeNow;
743}
744
745void Application::reshapeInternal(int width, int height) {
749 }
750 Application::application->reshape(width, height);
751}
752
753void Application::glfwOnChar(GLFWwindow* window, unsigned int key) {
754 if (Application::inputEventHandler == nullptr) return;
755 double mouseX, mouseY;
756 glfwGetCursorPos(window, &mouseX, &mouseY);
757 Application::inputEventHandler->onChar(key, (int)mouseX, (int)mouseY);
758}
759
760void Application::glfwOnKey(GLFWwindow* window, int key, int scanCode, int action, int mods) {
761 if (Application::inputEventHandler == nullptr) return;
762 double mouseX, mouseY;
763 glfwGetCursorPos(window, &mouseX, &mouseY);
764 // TODO: Use GLFW_MOD_CAPS_LOCK, which does not seem to be available with my version, need to update perhabs
765 if (key == GLFW_KEY_CAPS_LOCK) {
766 if (action == GLFW_PRESS) {
767 glfwCapsLockEnabled = glfwCapsLockEnabled == false?true:false;
768 }
769 }
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]),
774 key,
775 (int)mouseX,
776 (int)mouseY
777 );
778 } else
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]),
783 key,
784 (int)mouseX,
785 (int)mouseY
786 );
787 }
788}
789
790void Application::glfwOnMouseMoved(GLFWwindow* window, double x, double y) {
791 if (Application::inputEventHandler == nullptr) return;
794 } else {
796 }
797}
798
799void Application::glfwOnMouseButton(GLFWwindow* window, int button, int action, int mods) {
800 if (Application::inputEventHandler == nullptr) return;
801 double mouseX, mouseY;
802 glfwGetCursorPos(window, &mouseX, &mouseY);
803 Application::inputEventHandler->onMouseButton(button, action == GLFW_PRESS?MOUSE_BUTTON_DOWN:MOUSE_BUTTON_UP, (int)mouseX, (int)mouseY);
804 if (action == GLFW_PRESS) {
806 } else {
807 glfwMouseButtonDownFrames[button] = 0;
808 }
809 glfwMouseButtonLast = action == MOUSE_BUTTON_DOWN?button:-1;
810}
811
812void Application::glfwOnMouseWheel(GLFWwindow* window, double x, double y) {
813 if (Application::inputEventHandler == nullptr) return;
814 double mouseX, mouseY;
815 glfwGetCursorPos(window, &mouseX, &mouseY);
816 if (x != 0.0) Application::inputEventHandler->onMouseWheel(0, (int)x, (int)mouseX, (int)mouseY);
817 if (y != 0.0) Application::inputEventHandler->onMouseWheel(1, (int)y, (int)mouseX, (int)mouseY);
818}
819
820void Application::glfwOnWindowResize(GLFWwindow* window, int width, int height) {
821 Application::reshapeInternal(width, height);
822}
823
824void Application::glfwOnClose(GLFWwindow* window) {
826}
827
829}
static void glfwErrorCallback(int error, const char *description)
#define MOUSE_CURSOR_HAND
Definition: Application.h:14
#define MOUSE_CURSOR_ENABLED
Definition: Application.h:12
#define MOUSE_CURSOR_DISABLED
Definition: Application.h:11
#define MOUSE_BUTTON_DOWN
#define MOUSE_BUTTON_UP
Application base class, please make sure to allocate application on heap to have correct application ...
Definition: Application.h:37
static void setMousePosition(int x, int y)
Set mouse position.
static STATIC_DLL_IMPEXT Renderer * renderer
Definition: Application.h:313
static STATIC_DLL_IMPEXT int glfwMouseButtonLast
Definition: Application.h:332
static void glfwOnMouseWheel(GLFWwindow *window, double x, double y)
GLFW on key.
static void executeBackground(const string &command)
Execute a command in background.
static STATIC_DLL_IMPEXT Application * application
Definition: Application.h:314
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.
void setWindowXPosition(int windowXPosition)
Set window X position when initializing.
static void glfwOnClose(GLFWwindow *window)
GLFW on close.
static void glfwOnWindowResize(GLFWwindow *window, int width, int height)
GLFW on window resize.
static STATIC_DLL_IMPEXT array< unsigned int, 10 > glfwMouseButtonDownFrames
Definition: Application.h:331
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.
Definition: Application.cpp:91
void setClipboardContent(const string &content)
Set clipboard content.
static void displayInternal()
Display function.
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
Definition: Application.h:326
virtual ~Application()
Destructor.
static void cancelExit()
Cancels a users requested exit (ALT-F4 or X button)
virtual void onClose()
On close.
static STATIC_DLL_IMPEXT bool glfwCapsLockEnabled
Definition: Application.h:333
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
Definition: Application.h:325
void setFullScreen(bool fullScreen)
Set full screen mode.
static constexpr int WINDOW_HINT_MAXIMIZED
Definition: Application.h:47
static constexpr int FPS
Definition: Application.h:49
static STATIC_DLL_IMPEXT InputEventHandler * inputEventHandler
Definition: Application.h:315
virtual void initialize()=0
Init.
static constexpr int WINDOW_HINT_NOTDECORATED
Definition: Application.h:45
static constexpr int WINDOW_HINT_INVISIBLE
Definition: Application.h:46
static void exit(int exitCode)
Exits this application with given exit code.
static STATIC_DLL_IMPEXT GLFWwindow * glfwWindow
Definition: Application.h:330
static constexpr int WINDOW_HINT_NOTRESIZEABLE
Definition: Application.h:44
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
Definition: Application.h:334
Application input event handler interface.
virtual void onMouseWheel(int button, int direction, int x, int y)=0
On mouse wheen.
virtual void onKeyUp(unsigned char key, int keyCode, int x, int y)=0
On key up.
virtual void onMouseButton(int button, int state, int x, int y)=0
On mouse moved.
virtual void onMouseMoved(int x, int y)=0
On mouse moved.
virtual void onKeyDown(unsigned char key, int keyCode, int x, int y)=0
On key down.
virtual void onChar(unsigned int key, int x, int y)=0
On char.
virtual void onMouseDragged(int x, int y)=0
On mouse dragged.
Engine main class.
Definition: Engine.h:122
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.
Definition: FileSystem.h:14
Mutex implementation.
Definition: Mutex.h:27
Base class for threads.
Definition: Thread.h:26
Byte buffer class.
Definition: ByteBuffer.h:24
Character class.
Definition: Character.h:15
Console class.
Definition: Console.h:26
Integer to hex string conversion utility class.
Definition: Hex.h:16
Run time type information utility class.
Definition: RTTI.h:14
String tokenizer class.
void tokenize(const string &str, const string &delimiters)
Tokenize.
String tools class.
Definition: StringTools.h:20
Time utility class.
Definition: Time.h:21