TDME2 1.9.121
GUI.cpp
Go to the documentation of this file.
1#include <tdme/gui/GUI.h>
2
3#if defined(__APPLE__)
4 #include <Carbon/Carbon.h>
5#endif
6
7#include <algorithm>
8#include <string>
9#include <unordered_map>
10#include <unordered_set>
11#include <vector>
12
13#include <tdme/tdme.h>
16#include <tdme/engine/Engine.h>
35#include <tdme/utilities/Time.h>
36
37using std::remove;
38using std::string;
39using std::to_string;
40using std::unordered_map;
41using std::unordered_set;
42using std::vector;
43
59using tdme::gui::GUI;
67
68unordered_map<string, GUIFont*>* GUI::fontCache = new unordered_map<string, GUIFont*>();
69unordered_map<string, Texture*>* GUI::imageCache = new unordered_map<string, Texture*>();
70
71GUI::GUI(Engine* engine, GUIRenderer* guiRenderer)
72{
73 this->mouseButtonLast = 0;
74 this->engine = engine;
75 this->guiRenderer = guiRenderer;
76 this->width = engine->getWidth();
77 this->height = engine->getHeight();
78 try {
79 this->foccussedBorderColor = GUIColor("#5680C2");
80 } catch (Exception& exception) {
81 Console::print(string("GUI(): An error occurred: "));
82 Console::println(string(exception.what()));
83 }
84}
85
87}
88
90{
91}
92
93void GUI::reshape(int width, int height)
94{
95 this->width = width;
96 this->height = height;
97 for (auto screenIt: screens) {
98 auto screen = screenIt.second;
99 reshapeScreen(screen);
100 }
101}
102
104{
105 reset();
106}
107
108GUIFont* GUI::getFont(const string& applicationRootPath, const string& fileName)
109{
110 string canonicalFile;
111 string path;
112 string file;
113 GUIFont* font = nullptr;
114 try {
115 // TODO: improve me!
116 if (FileSystem::getInstance()->fileExists(fileName) == true) {
117 canonicalFile = fileName;
118 } else {
119 canonicalFile = FileSystem::getInstance()->getCanonicalPath(applicationRootPath, fileName);
120 }
121 path = FileSystem::getInstance()->getPathName(canonicalFile);
122 file = FileSystem::getInstance()->getFileName(canonicalFile);
123 } catch (Exception& exception) {
124 Console::print(string("GUI::getFont(): An error occurred: "));
125 Console::println(string(exception.what()));
126 return nullptr;
127 }
128
129 auto fontCacheIt = fontCache->find(canonicalFile);
130 if (fontCacheIt == fontCache->end()) {
131 try {
132 font = GUIFont::parse(path, file);
133 } catch (Exception& exception) {
134 Console::print(string("GUI::getFont(): An error occurred: "));
135 Console::println(string(exception.what()));
136 return nullptr;
137 }
138 (*fontCache)[canonicalFile] = font;
139 } else {
140 font = fontCacheIt->second;
141 }
142 return font;
143}
144
145Texture* GUI::getImage(const string& applicationRootPath, const string& fileName)
146{
147 // TODO: fix me, proper get path, filename
148 string canonicalFile;
149 string path;
150 string file;
151 try {
152 // TODO: improve me!
153 if (FileSystem::getInstance()->fileExists(fileName) == true) {
154 canonicalFile = fileName;
155 } else {
156 canonicalFile = FileSystem::getInstance()->getCanonicalPath(applicationRootPath, fileName);
157 }
158 path = FileSystem::getInstance()->getPathName(canonicalFile);
159 file = FileSystem::getInstance()->getFileName(canonicalFile);
160 } catch (Exception& exception) {
161 Console::print(string("GUI::getImage(): An error occurred: "));
162 Console::println(string(exception.what()));
163 return nullptr;
164 }
165
166 auto imageIt = imageCache->find("tdme.gui." + canonicalFile);
167 auto image = imageIt != imageCache->end() ? imageIt->second : nullptr;
168 if (image == nullptr) {
169 try {
170 image = TextureReader::read(path, file, false, false, "tdme.gui.");
171 if (image != nullptr) {
172 image->setUseMipMap(false);
173 image->setRepeat(false);
174 image->setClampMode(Texture::CLAMPMODE_TRANSPARENTPIXEL);
175 }
176 } catch (Exception& exception) {
177 Console::print(string("GUI::getImage(): An error occurred: "));
178 Console::println(string(exception.what()));
179 throw;
180 }
181 if (image != nullptr) (*imageCache)[canonicalFile] = image;
182 }
183 return image;
184}
185
186void GUI::addScreen(const string& id, GUIScreenNode* screen)
187{
188 if (id != screen->getId()) {
189 Console::println("GUI::addScreen(): screen id '" + id + "' must be '" + screen->getId() + "'");
190 return;
191 }
192 screens.emplace(id, screen);
193 reshapeScreen(screen);
194}
195
196void GUI::removeScreen(const string& id)
197{
198 auto screensIt = screens.find(id);
199 if (screensIt != screens.end()) {
201 auto screen = screensIt->second;
202 screens.erase(screensIt);
205 mousePressedEventNodeIds.erase(id);
207 mouseIsDragging.erase(id);
208 delete screen;
209 }
210}
211
213{
214 vector<string> entitiesToRemove;
215 for (auto screenKeysIt: screens) {
216 entitiesToRemove.push_back(screenKeysIt.first);
217 }
218 for (auto i = 0; i < entitiesToRemove.size(); i++) {
219 removeScreen(entitiesToRemove[i]);
220 }
221 renderScreens.clear();
222 /*
223 // TODO: fix me!
224 for (auto fontCacheIt: *fontCache) {
225 delete fontCacheIt.second;
226 }
227 fontCache->clear();
228 for (auto imageCacheIt: *imageCache) {
229 imageCacheIt.second->releaseReference();
230 }
231 imageCache->clear();
232 */
233}
234
236{
237 // focussed node
238 unfocusNode();
239 focussedNodeScreenId.clear();
240 focussedNodeNodeId.clear();
241
242 //
244
245 // unset GUI
246 for (auto i = 0; i < renderScreens.size(); i++) {
247 renderScreens[i]->setGUI(nullptr);
248 }
249
250 // clear
251 renderScreens.clear();
252}
253
254void GUI::addRenderScreen(const string& screenId)
255{
256 auto screenIt = screens.find(screenId);
257 if (screenIt == screens.end()) return;
258
259 //
261
262 //
263 auto screen = screenIt->second;
264 screen->setGUI(this);
265 screen->setConditionsMet();
266 renderScreens.push_back(screenIt->second);
267
268 // focussed node
269 focussedNodeScreenId.clear();
270 focussedNodeNodeId.clear();
272}
273
274void GUI::removeRenderScreen(const string& screenId)
275{
276 auto screenIt = screens.find(screenId);
277 if (screenIt == screens.end()) return;
278
279 //
280 screenIt->second->setGUI(nullptr);
281
282 //
284
285 //
286 renderScreens.erase(remove(renderScreens.begin(), renderScreens.end(), screenIt->second), renderScreens.end());
287}
288
290{
291 unfocusNode();
292 focussedNodeScreenId.clear();
293 focussedNodeNodeId.clear();
294}
295
297{
298 focusableNodes.clear();
299 focusableScreenNodes.clear();
300 for (int i = renderScreens.size() - 1; i >= 0; i--) {
301 auto screen = renderScreens[i];
302 if (screen->isVisible() == false)
303 continue;
304
305 focusableScreenNodes.push_back(screen);
306 if (screen->isPopUp() == true)
307 break;
308
309 }
310 for (int i = focusableScreenNodes.size() - 1; i >= 0; i--) {
311 auto screen = focusableScreenNodes[i];
312 screen->determineFocussedNodes(screen, focusableNodes);
313 }
314}
315
317{
318 if (focussedNodeScreenId.empty() == true || focussedNodeNodeId.empty() == true) return nullptr;
319 auto focussedNodeScreen = getScreen(focussedNodeScreenId);
320 auto focussedNode = dynamic_cast<GUIElementNode*>(focussedNodeScreen != nullptr?focussedNodeScreen->getNodeById(focussedNodeNodeId):nullptr);
321 return focussedNode;
322}
323
325{
326 if (focussedNodeScreenId.empty() == true || focussedNodeNodeId.empty() == true) return;
327 auto focussedNodeScreen = getScreen(focussedNodeScreenId);
328 auto focussedNode = dynamic_cast<GUIElementNode*>(focussedNodeScreen != nullptr?focussedNodeScreen->getNodeById(focussedNodeNodeId):nullptr);
329 if (focussedNode != nullptr) {
330 focussedNode->getActiveConditions().remove(GUIElementNode::CONDITION_FOCUS);
331 focussedNode->getBorder().topColor = unfocussedNodeBorderTopColor;
332 focussedNode->getBorder().leftColor = unfocussedNodeBorderLeftColor;
333 focussedNode->getBorder().bottomColor = unfocussedNodeBorderBottomColor;
334 focussedNode->getBorder().rightColor = unfocussedNodeBorderRightColor;
335 if (focussedNode->getController() != nullptr) focussedNode->getController()->onFocusLost();
336 focussedNode->getScreenNode()->delegateUnfocus(focussedNode);
337 }
338}
339
341{
342 auto focussedNodeScreen = getScreen(focussedNodeScreenId);
343 auto focussedNode = dynamic_cast<GUIElementNode*>(focussedNodeScreen != nullptr?focussedNodeScreen->getNodeById(focussedNodeNodeId):nullptr);
344 if (focussedNode != nullptr) {
345 focussedNode->getActiveConditions().add(GUIElementNode::CONDITION_FOCUS);
346 unfocussedNodeBorderTopColor = focussedNode->getBorder().topColor;
347 unfocussedNodeBorderLeftColor = focussedNode->getBorder().leftColor;
348 unfocussedNodeBorderBottomColor = focussedNode->getBorder().bottomColor;
349 unfocussedNodeBorderRightColor = focussedNode->getBorder().rightColor;
350 focussedNode->getBorder().topColor = foccussedBorderColor;
351 focussedNode->getBorder().leftColor = foccussedBorderColor;
352 focussedNode->getBorder().bottomColor = foccussedBorderColor;
353 focussedNode->getBorder().rightColor = foccussedBorderColor;
354 if (focussedNode->getController() != nullptr) focussedNode->getController()->onFocusGained();
355 focussedNode->getScreenNode()->delegateFocus(focussedNode);
356 }
357}
358
360{
361 auto focussedNodeScreen = getScreen(focussedNodeScreenId);
362 auto focussedNode = dynamic_cast<GUIElementNode*>(focussedNodeScreen != nullptr?focussedNodeScreen->getNodeById(focussedNodeNodeId):nullptr);
363 if (focussedNode == newFoccussedNode) return;
364 unfocusNode();
365 this->focussedNodeScreenId = newFoccussedNode == nullptr?string():newFoccussedNode->getScreenNode()->getId();
366 this->focussedNodeNodeId = newFoccussedNode == nullptr?string():newFoccussedNode->getId();
367 if (newFoccussedNode != nullptr) focusNode();
369}
370
372{
374 unfocusNode();
375 auto focussedNodeScreen = getScreen(focussedNodeScreenId);
376 auto focussedNode = dynamic_cast<GUIElementNode*>(focussedNodeScreen != nullptr?focussedNodeScreen->getNodeById(focussedNodeNodeId):nullptr);
377 if (focusableNodes.size() > 0) {
378 auto focussedNodeIdx = -1;
379 for (auto i = 0; i < focusableNodes.size(); i++) {
380 if (focussedNode == focusableNodes[i]) {
381 focussedNodeIdx = i;
382 }
383 }
384 auto focussedNextNodeIdx = (focussedNodeIdx + 1) % focusableNodes.size();
385 auto newFocussedNode = focusableNodes[focussedNextNodeIdx];
386 setFoccussedNode(newFocussedNode);
387 newFocussedNode->scrollToNodeX();
388 newFocussedNode->scrollToNodeY();
389 }
390}
391
393{
395 unfocusNode();
396 auto focussedNodeScreen = getScreen(focussedNodeScreenId);
397 auto focussedNode = dynamic_cast<GUIElementNode*>(focussedNodeScreen != nullptr?focussedNodeScreen->getNodeById(focussedNodeNodeId):nullptr);
398 if (focusableNodes.size() > 0) {
399 auto focussedNodeIdx = -1;
400 for (auto i = 0; i < focusableNodes.size(); i++) {
401 if (focussedNode == focusableNodes[i]) {
402 focussedNodeIdx = i;
403 }
404 }
405 int focussedPreviousNodeIdx = (focussedNodeIdx - 1) % focusableNodes.size();
406 if (focussedPreviousNodeIdx < 0) focussedPreviousNodeIdx += focusableNodes.size();
407 auto newFocussedNode = focusableNodes[focussedPreviousNodeIdx];
408 setFoccussedNode(newFocussedNode);
409 newFocussedNode->scrollToNodeX();
410 newFocussedNode->scrollToNodeY();
411 }
412}
413
415{
416 //
417 if (renderScreens.empty() == true)
418 return;
419
420 // invalidate layouts
421 for (int i = renderScreens.size() - 1; i >= 0; i--) {
422 auto screen = renderScreens[i];
423
424 if (screen->isVisible() == false) continue;
425
426 screen->invalidateLayouts();
427 screen->scrollToNodes();
428 }
429
430 //
431 if (focussedNodeScreenId.empty() == true || focussedNodeNodeId.empty() == true) {
433 }
434
435 guiRenderer->setGUI(this);
438 for (auto i = 0; i < renderScreens.size(); i++) {
439 auto screen = renderScreens[i];
440
441 if (screen->isVisible() == false) continue;
442
443 screen->render(guiRenderer);
444 }
445 for (auto i = 0; i < renderScreens.size(); i++) {
446 auto screen = renderScreens[i];
447 if (screen->isVisible() == false) continue;
448
449 screen->renderFloatingNodes(guiRenderer);
450 }
453}
454
456 auto nodeId = node->getId();
457 auto screenNodeId = node->getScreenNode()->getId();
458 return
459 mousePressedEventNodeIds[screenNodeId].find(nodeId) != mousePressedEventNodeIds[screenNodeId].end() ||
460 mouseDraggingEventNodeIds[screenNodeId].find(nodeId) != mouseDraggingEventNodeIds[screenNodeId].end();
461}
462
463void GUI::handleMouseEvent(GUINode* node, GUIMouseEvent* event, const unordered_set<string>& mouseOutCandidateEventNodeIds, const unordered_set<string>& mouseOutClickCandidateEventNodeIds, unordered_set<string>& mousePressedEventNodeIds, bool floatingNodes)
464{
465 // handle each event
466 unordered_set<string> mouseEventNodeIgnore;
467 unordered_set<string> mouseEventNodeIds;
468
469 //
470 event->setX((float)event->getXUnscaled() * (float)node->getScreenNode()->getScreenWidth() / (float)width + node->getScreenNode()->getGUIEffectOffsetX());
471 event->setY((float)event->getYUnscaled() * (float)node->getScreenNode()->getScreenHeight() / (float)height + node->getScreenNode()->getGUIEffectOffsetY());
472
473 // if dragging only send events to dragging origin nodes
474 if (event->getType() == GUIMouseEvent::MOUSEEVENT_DRAGGED) {
475 mouseEventNodeIds = mouseDraggingEventNodeIds[node->getScreenNode()->getId()];
476 } else
477 if (event->getType() == GUIMouseEvent::MOUSEEVENT_RELEASED &&
478 mouseIsDragging[node->getScreenNode()->getId()] == true) {
479 mouseEventNodeIds = mouseDraggingEventNodeIds[node->getScreenNode()->getId()];
480 } else {
481 // otherwise continue with determining nodes that receive this event
482 if (floatingNodes == true) {
483 node->determineMouseEventNodes(event, node->flow == GUINode_Flow::FLOATING, mouseEventNodeIgnore, mouseEventNodeIds);
484 } else {
485 node->determineMouseEventNodes(event, node->flow == GUINode_Flow::FLOATING, mouseEventNodeIds, mouseEventNodeIgnore);
486 }
487 }
488
489 // inject former mouse out candidates
490 for (auto eventNodeId: mouseOutCandidateEventNodeIds) {
491 mouseEventNodeIds.insert(eventNodeId);
492 }
493
494 // inject former mouse out click candidates
495 for (auto eventNodeId: mouseOutClickCandidateEventNodeIds) {
496 mouseEventNodeIds.insert(eventNodeId);
497 }
498
499 // handle mouse event for each node we collected
500 for (auto eventNodeId: mouseEventNodeIds) {
501 // node event occurred on
502 auto eventNode = node->getScreenNode()->getNodeById(eventNodeId);
503 if (eventNode == nullptr) continue;
504 if (floatingNodes == false && eventNode->flow == GUINode_Flow::FLOATING) continue;
505
506 // controller node
507 auto controllerNode = eventNode;
508 if (controllerNode->getController() == nullptr) {
509 controllerNode = controllerNode->getParentControllerNode();
510 }
511 if (controllerNode == nullptr) continue;
512
513 // this corresponding screen node could have been detached from GUI by removing it from render screens
514 if (eventNode->getScreenNode()->getGUI() == nullptr) continue;
515
516 // handle event with determined controller
517 controllerNode->getController()->handleMouseEvent(eventNode, event);
518 }
519
520 // compile list of gui node ids that received mouse PRESSED events
521 if (event->getType() == GUIMouseEvent::MOUSEEVENT_PRESSED) {
522 for (auto eventNodeId: mouseEventNodeIds) {
523 mousePressedEventNodeIds.insert(eventNodeId);
524 }
525 }
526
527 // clear event node id list
528 mouseEventNodeIds.clear();
529}
530
532 // skip if already processed
533 if (event->isProcessed() == true) return;
534
535 //
536 switch (event->getKeyCode()) {
537 case (GUIKeyboardEvent::KEYCODE_TAB):
538 {
539 if (event->getType() == GUIKeyboardEvent::KEYBOARDEVENT_KEY_PRESSED) {
541 }
542 event->setProcessed(true);
543 break;
544 }
545 case (GUIKeyboardEvent::KEYCODE_TAB_SHIFT):
546 {
547 if (event->getType() == GUIKeyboardEvent::KEYBOARDEVENT_KEY_PRESSED) {
549 }
550 event->setProcessed(true);
551 break;
552 }
553 case (GUIKeyboardEvent::KEYCODE_LEFT_ALT):
554 case (GUIKeyboardEvent::KEYCODE_RIGHT_ALT):
555 {
556 altDown = event->getType() == GUIKeyboardEvent::KEYBOARDEVENT_KEY_PRESSED;
557 break;
558 }
559 case (GUIKeyboardEvent::KEYCODE_LEFT_CONTROL):
560 case (GUIKeyboardEvent::KEYCODE_RIGHT_CONTROL):
561 {
562 controlDown = event->getType() == GUIKeyboardEvent::KEYBOARDEVENT_KEY_PRESSED;
563 break;
564 }
565 case (GUIKeyboardEvent::KEYCODE_LEFT_SHIFT):
566 case (GUIKeyboardEvent::KEYCODE_RIGHT_SHIFT):
567 {
568 shiftDown = event->getType() == GUIKeyboardEvent::KEYBOARDEVENT_KEY_PRESSED;
569 break;
570 }
571 default:
572 {
573 break;
574 }
575 }
576
577 // skip if processed
578 if (event->isProcessed() == true) return;
579
580 // otherwise dispatch event to focussed node
581 if (focussedNodeScreenId.empty() == false && focussedNodeNodeId.empty() == false) {
582 auto focussedNodeScreen = getScreen(focussedNodeScreenId);
583 auto focussedNode = dynamic_cast<GUIElementNode*>(focussedNodeScreen != nullptr?focussedNodeScreen->getNodeById(focussedNodeNodeId):nullptr);
584 if (focussedNode != nullptr) focussedNode->handleKeyboardEvent(event);
585 }
586}
587
589{
590 unordered_map<string, unordered_set<string>> _mouseOutCandidateEventNodeIds;
591 unordered_map<string, unordered_set<string>> _mouseOutClickCandidateEventNodeIds;
592
593 //
594 auto renderScreensCopy = renderScreens;
595
596 // handle mouse events
597 for (auto& event: mouseEvents) {
598
599 // fetch and clear last mouse out candidates as we now will actually send them
600 if (event.getType() == GUIMouseEvent::MOUSEEVENT_MOVED) {
601 _mouseOutCandidateEventNodeIds = mouseOutCandidateEventNodeIds;
603 } else
604 if (event.getType() == GUIMouseEvent::MOUSEEVENT_PRESSED || event.getType() == GUIMouseEvent::MOUSEEVENT_RELEASED) {
605 _mouseOutClickCandidateEventNodeIds = mouseOutClickCandidateEventNodeIds;
607 }
608
609 // handle mouse dragged event
610 if (event.getType() == GUIMouseEvent::MOUSEEVENT_DRAGGED) {
611 for (int i = renderScreensCopy.size() - 1; i >= 0; i--) {
612 auto screen = renderScreensCopy[i];
613 if (mouseIsDragging[screen->getId()] == false) {
614 mouseIsDragging[screen->getId()] = true;
615 mouseDraggingEventNodeIds[screen->getId()] = mousePressedEventNodeIds[screen->getId()];
616 }
617 }
618 }
619
620 // handle floating nodes first
621 for (int i = renderScreensCopy.size() - 1; i >= 0; i--) {
622 auto screen = renderScreensCopy[i];
623
624 // skip on invisible
625 if (screen->isVisible() == false) continue;
626
627 //
628 auto processed = false;
629 auto& floatingNodes = screen->getFloatingNodes();
630 for (auto j = 0; j < floatingNodes.size(); j++) {
631 auto floatingNode = floatingNodes.at(j);
632
633 handleMouseEvent(floatingNode, &event, _mouseOutCandidateEventNodeIds[screen->getId()], _mouseOutClickCandidateEventNodeIds[screen->getId()], mousePressedEventNodeIds[screen->getId()], true);
634
635 // do not continue handling mouse events if already processed
636 if (event.isProcessed() == true) break;
637
638 // We do not consume mouse moved events by default, so:
639 // consume mouse move event, if floating node has conditions met and is expanded/collapsed by condition
640 // as in this case you do not want to have other mouse movement event handling
641 if ((event.getType() == GUIMouseEvent::MOUSEEVENT_MOVED || floatingNode->isEventBelongingToNode(&event) == true) &&
642 floatingNode->flow == GUINode_Flow::FLOATING &&
643 floatingNode->conditionsMet == true &&
644 ((floatingNode->showOn.getConditions().empty() == false && floatingNode->showOn.has("always") == false) ||
645 (floatingNode->hideOn.getConditions().empty() == false && floatingNode->showOn.has("never") == false))) processed = true;
646 }
647 // do not continue handling mouse events if already processed
648 if (event.isProcessed() == true) break;
649 // set processed if we decided to do so :D
650 if (processed == true) event.setProcessed(true);
651 // skip on pop ups
652 if (screen->isPopUp() == true) break;
653 }
654
655 // handle normal screen nodes if not processed already by floating node
656 // Note: Different screens should not have UI elements that overlap and process events
657 if (event.isProcessed() == false) {
658 for (int i = renderScreensCopy.size() - 1; i >= 0; i--) {
659 auto screen = renderScreensCopy[i];
660 if (screen->isVisible() == false) continue;
661 handleMouseEvent(screen, &event, _mouseOutCandidateEventNodeIds[screen->getId()], _mouseOutClickCandidateEventNodeIds[screen->getId()], mousePressedEventNodeIds[screen->getId()], false);
662 if (screen->isPopUp() == true) break;
663 }
664 }
665
666 // handle mouse released event
667 if (event.getType() == GUIMouseEvent::MOUSEEVENT_RELEASED) {
668 for (int i = renderScreensCopy.size() - 1; i >= 0; i--) {
669 auto screen = renderScreensCopy[i];
670 mouseIsDragging[screen->getId()] = false;
671 mouseDraggingEventNodeIds.erase(screen->getId());
672 mousePressedEventNodeIds.erase(screen->getId());
673 }
674 }
675 }
676
677 // handle keyboard events
678 for (auto& event: keyboardEvents) {
679 handleKeyboardEvent(&event);
680 }
681
682 // call tick and input event handler at very last
683 for (int i = renderScreensCopy.size() - 1; i >= 0; i--) {
684 auto screen = renderScreensCopy[i];
685 if (screen->isVisible() == false) continue;
686 screen->tick();
687 if (screen->getInputEventHandler() != nullptr) {
688 screen->getInputEventHandler()->handleInputEvents();
689 }
690 if (screen->isPopUp() == true) break;
691 }
692
693 //
694 mouseEvents.clear();
695 keyboardEvents.clear();
696}
697
698void GUI::onChar(unsigned int key, int x, int y) {
700 GUIKeyboardEvent guiKeyboardEvent;
701 guiKeyboardEvent.setTime(Time::getCurrentMillis());
702 guiKeyboardEvent.setType(GUIKeyboardEvent::KEYBOARDEVENT_KEY_TYPED);
703 guiKeyboardEvent.setKeyCode(-1);
704 guiKeyboardEvent.setKeyChar(key);
705 guiKeyboardEvent.setMetaDown(false);
706 guiKeyboardEvent.setControlDown(controlDown);
707 guiKeyboardEvent.setAltDown(altDown);
708 guiKeyboardEvent.setShiftDown(shiftDown);
709 guiKeyboardEvent.setProcessed(false);
710 keyboardEvents.push_back(guiKeyboardEvent);
711}
712
713void GUI::onKeyDown (unsigned char key, int keyCode, int x, int y) {
715 GUIKeyboardEvent guiKeyboardEvent;
716 guiKeyboardEvent.setTime(Time::getCurrentMillis());
717 guiKeyboardEvent.setType(GUIKeyboardEvent::KEYBOARDEVENT_KEY_PRESSED);
718 guiKeyboardEvent.setKeyCode(keyCode);
719 guiKeyboardEvent.setKeyChar(key);
720 guiKeyboardEvent.setMetaDown(false);
721 guiKeyboardEvent.setControlDown(controlDown);
722 guiKeyboardEvent.setAltDown(altDown);
723 guiKeyboardEvent.setShiftDown(shiftDown);
724 guiKeyboardEvent.setProcessed(false);
725 keyboardEvents.push_back(guiKeyboardEvent);
726}
727
728void GUI::onKeyUp(unsigned char key, int keyCode, int x, int y) {
730 GUIKeyboardEvent guiKeyboardEvent;
731 guiKeyboardEvent.setTime(Time::getCurrentMillis());
732 guiKeyboardEvent.setType(GUIKeyboardEvent::KEYBOARDEVENT_KEY_RELEASED);
733 guiKeyboardEvent.setKeyCode(keyCode);
734 guiKeyboardEvent.setKeyChar(key);
735 guiKeyboardEvent.setMetaDown(false);
736 guiKeyboardEvent.setControlDown(controlDown);
737 guiKeyboardEvent.setAltDown(altDown);
738 guiKeyboardEvent.setShiftDown(shiftDown);
739 guiKeyboardEvent.setProcessed(false);
740 keyboardEvents.push_back(guiKeyboardEvent);
741}
742
743void GUI::onMouseDragged(int x, int y) {
745
746 GUIMouseEvent guiMouseEvent;
747 guiMouseEvent.setTime(Time::getCurrentMillis());
748 guiMouseEvent.setType(GUIMouseEvent::MOUSEEVENT_DRAGGED);
749 guiMouseEvent.setXUnscaled(x);
750 guiMouseEvent.setYUnscaled(y);
751 guiMouseEvent.setX(x);
752 guiMouseEvent.setY(y);
753 guiMouseEvent.setButton(mouseButtonLast);
754 guiMouseEvent.setWheelX(0.0f);
755 guiMouseEvent.setWheelY(0.0f);
756 guiMouseEvent.setWheelZ(0.0f);
757 guiMouseEvent.setControlDown(controlDown);
758 guiMouseEvent.setAltDown(altDown);
759 guiMouseEvent.setShiftDown(shiftDown);
760 guiMouseEvent.setProcessed(false);
761 mouseEvents.push_back(guiMouseEvent);
762}
763
764void GUI::onMouseMoved(int x, int y) {
766
767 GUIMouseEvent guiMouseEvent;
768 guiMouseEvent.setTime(Time::getCurrentMillis());
769 guiMouseEvent.setType(GUIMouseEvent::MOUSEEVENT_MOVED);
770 guiMouseEvent.setXUnscaled(x);
771 guiMouseEvent.setYUnscaled(y);
772 guiMouseEvent.setX(x);
773 guiMouseEvent.setY(y);
774 guiMouseEvent.setButton(0);
775 guiMouseEvent.setWheelX(0.0f);
776 guiMouseEvent.setWheelY(0.0f);
777 guiMouseEvent.setWheelZ(0.0f);
778 guiMouseEvent.setControlDown(controlDown);
779 guiMouseEvent.setAltDown(altDown);
780 guiMouseEvent.setShiftDown(shiftDown);
781 guiMouseEvent.setProcessed(false);
782 mouseEvents.push_back(guiMouseEvent);
783}
784
785void GUI::onMouseButton(int button, int state, int x, int y) {
787
788 mouseButtonLast = button + 1;
789 GUIMouseEvent guiMouseEvent;
790 guiMouseEvent.setTime(Time::getCurrentMillis());
791 guiMouseEvent.setType(state == MOUSE_BUTTON_DOWN?GUIMouseEvent::MOUSEEVENT_PRESSED:GUIMouseEvent::MOUSEEVENT_RELEASED);
792 guiMouseEvent.setXUnscaled(x);
793 guiMouseEvent.setYUnscaled(y);
794 guiMouseEvent.setX(x);
795 guiMouseEvent.setY(y);
796 guiMouseEvent.setButton(mouseButtonLast);
797 guiMouseEvent.setWheelX(0.0f);
798 guiMouseEvent.setWheelY(0.0f);
799 guiMouseEvent.setWheelZ(0.0f);
800 guiMouseEvent.setControlDown(controlDown);
801 guiMouseEvent.setAltDown(altDown);
802 guiMouseEvent.setShiftDown(shiftDown);
803 guiMouseEvent.setProcessed(false);
804 mouseEvents.push_back(guiMouseEvent);
805}
806
807void GUI::onMouseWheel(int button, int direction, int x, int y) {
809
810 mouseButtonLast = button + 1;
811 GUIMouseEvent guiMouseEvent;
812 guiMouseEvent.setTime(Time::getCurrentMillis());
813 guiMouseEvent.setType(GUIMouseEvent::MOUSEEVENT_WHEEL_MOVED);
814 guiMouseEvent.setXUnscaled(x);
815 guiMouseEvent.setYUnscaled(y);
816 guiMouseEvent.setX(x);
817 guiMouseEvent.setY(y);
818 guiMouseEvent.setButton(mouseButtonLast);
819 guiMouseEvent.setWheelX(0.0f);
820 guiMouseEvent.setWheelY(direction * 1.0f);
821 guiMouseEvent.setWheelZ(0.0f);
822 guiMouseEvent.setControlDown(controlDown);
823 guiMouseEvent.setAltDown(altDown);
824 guiMouseEvent.setShiftDown(shiftDown);
825 guiMouseEvent.setProcessed(false);
826 mouseEvents.push_back(guiMouseEvent);
827}
828
830{
831 GUIMouseEvent guiMouseEvent;
832 guiMouseEvent.setTime(Time::getCurrentMillis());
833 guiMouseEvent.setType(GUIMouseEvent::MOUSEEVENT_MOVED);
834 guiMouseEvent.setXUnscaled(-10000);
835 guiMouseEvent.setYUnscaled(-10000);
836 guiMouseEvent.setX(-10000);
837 guiMouseEvent.setY(-10000);
838 guiMouseEvent.setButton(0);
839 guiMouseEvent.setWheelX(0.0f);
840 guiMouseEvent.setWheelY(0.0f);
841 guiMouseEvent.setWheelZ(0.0f);
842 guiMouseEvent.setProcessed(false);
843 mouseEvents.push_back(guiMouseEvent);
844}
845
847 bool isControlDown = false;
848 bool isAltDown = false;
849 bool isShiftDown = false;
850 #if defined(__APPLE__)
851 KeyMap keys;
852 GetKeys(keys);
853 #define IS_KEY_DOWN(key, var) \
854 { \
855 uint8_t index = key / 32 ; \
856 uint8_t shift = key % 32 ; \
857 var = keys[index].bigEndianValue & (1 << shift); \
858 }
859 IS_KEY_DOWN(kVK_Command, isControlDown);
860 IS_KEY_DOWN(kVK_Option, isAltDown);
861 IS_KEY_DOWN(kVK_Shift, isShiftDown);
862 #endif
863
864 if (isControlDown == false &&
865 isAltDown == false &&
866 isShiftDown == false) {
867 return;
868 }
869
870 GUIKeyboardEvent guiKeyboardEvent;
871 guiKeyboardEvent.setTime(Time::getCurrentMillis());
872 guiKeyboardEvent.setType(GUIKeyboardEvent::KEYBOARDEVENT_KEY_PRESSED);
873 guiKeyboardEvent.setKeyCode(-1);
874 guiKeyboardEvent.setKeyChar(-1);
875 guiKeyboardEvent.setMetaDown(false);
876 guiKeyboardEvent.setControlDown(isControlDown);
877 guiKeyboardEvent.setAltDown(isAltDown);
878 guiKeyboardEvent.setShiftDown(isShiftDown);
879 guiKeyboardEvent.setProcessed(false);
880 keyboardEvents.push_back(guiKeyboardEvent);
881}
882
884 // TODO: maybe move logic into GUIScreenNode
885 // FIXME: always keep aspect ratio
886 auto screenNodeWidthConstrained = width;
887 auto screenNodeHeightConstrained = height;
888
889 auto minRatio = 1.0f;
890
891 if ((screenNode->sizeConstraints.maxWidth > 0 && screenNodeWidthConstrained > screenNode->sizeConstraints.maxWidth) ||
892 (screenNode->sizeConstraints.maxHeight > 0 && screenNodeHeightConstrained > screenNode->sizeConstraints.maxHeight)) {
893 if (screenNode->sizeConstraints.maxWidth > 0 && screenNodeWidthConstrained > screenNode->sizeConstraints.maxWidth) {
894 minRatio = (float)screenNode->sizeConstraints.maxWidth / (float)width;
895 screenNodeWidthConstrained = screenNode->sizeConstraints.maxWidth;
896 screenNodeHeightConstrained = (int)((float)screenNodeHeightConstrained * minRatio);
897 }
898 if (screenNode->sizeConstraints.maxHeight > 0 && screenNodeHeightConstrained > screenNode->sizeConstraints.maxHeight) {
899 minRatio = (float)screenNode->sizeConstraints.maxHeight / (float)height;
900 screenNodeHeightConstrained = screenNode->sizeConstraints.maxHeight;
901 screenNodeWidthConstrained = (int)((float)screenNodeWidthConstrained * minRatio);
902 }
903 }
904 if ((screenNode->sizeConstraints.minWidth > 0 && screenNodeWidthConstrained < screenNode->sizeConstraints.minWidth) ||
905 (screenNode->sizeConstraints.minHeight > 0 && screenNodeHeightConstrained < screenNode->sizeConstraints.minHeight)) {
906 if (screenNode->sizeConstraints.minWidth > 0 && screenNodeWidthConstrained < screenNode->sizeConstraints.minWidth) {
907 auto ratio = (float)screenNode->sizeConstraints.minWidth / (float)width;
908 screenNodeWidthConstrained = screenNode->sizeConstraints.minWidth;
909 screenNodeHeightConstrained = (int)((float)screenNodeHeightConstrained * ratio / minRatio);
910 }
911 if (screenNode->sizeConstraints.minHeight > 0 && screenNodeHeightConstrained < screenNode->sizeConstraints.minHeight) {
912 auto ratio = (float)screenNode->sizeConstraints.minHeight / (float)height;
913 screenNodeHeightConstrained = screenNode->sizeConstraints.minHeight;
914 screenNodeWidthConstrained = (int)((float)screenNodeWidthConstrained * ratio / minRatio);
915 }
916 }
917
918 screenNode->setScreenSize(screenNodeWidthConstrained, screenNodeHeightConstrained);
919}
920
922 for (auto screen: renderScreens) screen->unsetMouseOver();
923}
#define MOUSE_BUTTON_DOWN
Engine main class.
Definition: Engine.h:122
int32_t getWidth()
Definition: Engine.h:865
void doneGUIMode()
Set up GUI mode rendering.
Definition: Engine.cpp:1962
void initGUIMode()
Set up GUI mode rendering.
Definition: Engine.cpp:1950
int32_t getHeight()
Definition: Engine.h:872
GUI module class.
Definition: GUI.h:66
unordered_map< string, unordered_set< string > > mousePressedEventNodeIds
Definition: GUI.h:98
GUIColor unfocussedNodeBorderTopColor
Definition: GUI.h:88
void onMouseMoved(int x, int y) override
On mouse moved.
Definition: GUI.cpp:764
Engine * engine
Definition: GUI.h:79
bool controlDown
Definition: GUI.h:103
bool shiftDown
Definition: GUI.h:104
GUIElementNode * getFocussedNode()
Definition: GUI.cpp:316
vector< GUIScreenNode * > renderScreens
Definition: GUI.h:92
string focussedNodeScreenId
Definition: GUI.h:84
bool altDown
Definition: GUI.h:102
unordered_map< string, unordered_set< string > > mouseOutCandidateEventNodeIds
Definition: GUI.h:96
static Texture * getImage(const string &applicationRootPath, const string &fileName)
Get image.
Definition: GUI.cpp:145
void onMouseDragged(int x, int y) override
On mouse dragged.
Definition: GUI.cpp:743
int width
Definition: GUI.h:93
void initialize()
Init.
Definition: GUI.cpp:89
void focusPreviousNode()
Focus next node.
Definition: GUI.cpp:392
bool isHavingMouseInteraction(GUINode *node)
Reports if node has currently mouse interaction like dragging or pressing.
Definition: GUI.cpp:455
void applyRenderScreensChange()
Render screens change.
Definition: GUI.cpp:921
void removeRenderScreen(const string &screenId)
Remove render screen.
Definition: GUI.cpp:274
~GUI()
Destructor.
Definition: GUI.cpp:86
vector< GUIScreenNode * > focusableScreenNodes
Definition: GUI.h:83
vector< GUIElementNode * > focusableNodes
Definition: GUI.h:82
void onMouseButton(int button, int state, int x, int y) override
On mouse moved.
Definition: GUI.cpp:785
void handleMouseEvent(GUINode *node, GUIMouseEvent *event, const unordered_set< string > &mouseOutCandidateEventNodeIds, const unordered_set< string > &mouseOutClickCandidateEventNodeIds, unordered_set< string > &mousePressedEventNodeIds, bool floatingNodes)
Handle mouse event for given node.
Definition: GUI.cpp:463
unordered_map< string, bool > mouseIsDragging
Definition: GUI.h:100
void determineFocussedNodes()
Determine focussed nodes.
Definition: GUI.cpp:296
void onMouseWheel(int button, int direction, int x, int y) override
On mouse wheel.
Definition: GUI.cpp:807
GUIColor foccussedBorderColor
Definition: GUI.h:81
int mouseButtonLast
Definition: GUI.h:95
unordered_map< string, GUIScreenNode * > screens
Definition: GUI.h:80
GUIRenderer * guiRenderer
Definition: GUI.h:78
static STATIC_DLL_IMPEXT unordered_map< string, Texture * > * imageCache
Definition: GUI.h:76
void handleEvents()
Handle screen events.
Definition: GUI.cpp:588
void reshape(int width, int height)
Reshape.
Definition: GUI.cpp:93
static STATIC_DLL_IMPEXT unordered_map< string, GUIFont * > * fontCache
Definition: GUI.h:75
string focussedNodeNodeId
Definition: GUI.h:85
void unfocusNode()
Unfocus current focussed node.
Definition: GUI.cpp:324
void render()
Render GUIs.
Definition: GUI.cpp:414
void onKeyUp(unsigned char key, int keyCode, int x, int y) override
On key up.
Definition: GUI.cpp:728
void onKeyDown(unsigned char key, int keyCode, int x, int y) override
On key down.
Definition: GUI.cpp:713
vector< GUIKeyboardEvent > keyboardEvents
Definition: GUI.h:91
void setFoccussedNode(GUIElementNode *newFoccussedNode)
Set focussed node.
Definition: GUI.cpp:359
unordered_map< string, unordered_set< string > > mouseOutClickCandidateEventNodeIds
Definition: GUI.h:97
void focusNode()
Focus current focussed node.
Definition: GUI.cpp:340
void fakeKeyboardModifierEvent()
Fake a keyboard modifier event.
Definition: GUI.cpp:846
GUIColor unfocussedNodeBorderRightColor
Definition: GUI.h:87
GUIScreenNode * getScreen(const string &id)
Get screen.
Definition: GUI.h:246
void addRenderScreen(const string &screenId)
Add render screen.
Definition: GUI.cpp:254
void removeScreen(const string &id)
Removes an screen.
Definition: GUI.cpp:196
void addScreen(const string &id, GUIScreenNode *screen)
Add screen.
Definition: GUI.cpp:186
void dispose()
Dispose.
Definition: GUI.cpp:103
void invalidateFocussedNode()
Invalidate focussed node.
Definition: GUI.cpp:289
void reshapeScreen(GUIScreenNode *screenNode)
Reshape screen.
Definition: GUI.cpp:883
unordered_map< string, unordered_set< string > > mouseDraggingEventNodeIds
Definition: GUI.h:99
void fakeMouseMovedEvent()
Fake mouse moved event.
Definition: GUI.cpp:829
void focusNextNode()
Focus next node.
Definition: GUI.cpp:371
GUIColor unfocussedNodeBorderLeftColor
Definition: GUI.h:86
int height
Definition: GUI.h:94
void reset()
Removes all screens and caches.
Definition: GUI.cpp:212
void handleKeyboardEvent(GUIKeyboardEvent *event)
Handle mouse event for given node.
Definition: GUI.cpp:531
static GUIFont * getFont(const string &applicationRootPath, const string &fileName)
Get font.
Definition: GUI.cpp:108
void resetRenderScreens()
Reset render screens.
Definition: GUI.cpp:235
void onChar(unsigned int key, int x, int y) override
On char.
Definition: GUI.cpp:698
GUIColor unfocussedNodeBorderBottomColor
Definition: GUI.h:89
vector< GUIMouseEvent > mouseEvents
Definition: GUI.h:90
void setKeyCode(int32_t code)
Set key code.
void setAltDown(bool altDown)
Set alt down.
void setProcessed(bool processed)
Set event processed.
void setType(GUIKeyboardEventType type)
Set type.
void setControlDown(bool controlDown)
Set control down.
void setKeyChar(char keyChar)
Set key char.
void setTime(int64_t time)
Time in milliseconds.
void setMetaDown(bool metaDown)
Set meta down.
void setShiftDown(bool shiftDown)
Set shift down.
void setYUnscaled(int yUnscaled)
Set y unscaled.
void setAltDown(bool altDown)
Set alt down.
void setWheelX(float wheelX)
Set up wheel x.
void setProcessed(bool processed)
Set processed.
void setControlDown(bool controlDown)
Set control down.
void setWheelZ(float wheelZ)
Set up wheel z.
void setXUnscaled(int xUnscaled)
Set x unscaled.
Definition: GUIMouseEvent.h:88
void setTime(int64_t time)
Set time.
Definition: GUIMouseEvent.h:58
void setButton(int button)
Set button.
void setShiftDown(bool shiftDown)
Set shift down.
void setType(GUIMouseEventType type)
Set type.
Definition: GUIMouseEvent.h:73
void setWheelY(float wheelY)
Set up wheel y.
void handleKeyboardEvent(GUIKeyboardEvent *event)
Handle keyboard event.
GUINodeConditions & getActiveConditions()
GUI element node conditions.
bool add(const string &condition)
Add a condition.
bool remove(const string &condition)
Remove a condition.
GUI node controller base class.
virtual void handleMouseEvent(GUINode *node, GUIMouseEvent *event)=0
Handle mouse event.
GUI node base class.
Definition: GUINode.h:63
virtual void determineMouseEventNodes(GUIMouseEvent *event, bool floatingNode, unordered_set< string > &eventNodeIds, unordered_set< string > &eventFloatingNodeIds)
Determine mouse event nodes.
Definition: GUINode.cpp:1062
GUIParentNode * getParentControllerNode()
Definition: GUINode.cpp:1053
const string & getId()
Definition: GUINode.h:329
GUIScreenNode * getScreenNode()
Definition: GUINode.h:315
GUINode_Flow * flow
Definition: GUINode.h:85
GUINodeController * getController()
Definition: GUINode.h:586
GUI screen node that represents a screen that can be rendered via GUI system.
Definition: GUIScreenNode.h:57
GUINode * getNodeById(const string &nodeId)
Get GUI node by id.
GUIScreenNode_SizeConstraints sizeConstraints
Definition: GUIScreenNode.h:83
void setScreenSize(int width, int height)
Set screen size.
GUI Font A font implementation that will parse the output of the AngelCode font tool available at:
Definition: GUIFont.h:48
void initRendering()
Init rendering.
void doneRendering()
Done rendering.
void setGUI(GUI *gui)
Set GUI.
Definition: GUIRenderer.h:101
File system singleton class.
Definition: FileSystem.h:14
Console class.
Definition: Console.h:26
Time utility class.
Definition: Time.h:21
std::exception Exception
Exception base class.
Definition: Exception.h:19
GUI input event handler interface.
GUI node border entity.