BsGUIManager.cpp 46 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482
  1. #include "BsGUIManager.h"
  2. #include "BsGUIWidget.h"
  3. #include "BsGUIElement.h"
  4. #include "BsImageSprite.h"
  5. #include "BsSpriteTexture.h"
  6. #include "BsTime.h"
  7. #include "BsSceneObject.h"
  8. #include "BsMaterial.h"
  9. #include "BsMeshData.h"
  10. #include "BsVertexDataDesc.h"
  11. #include "BsMesh.h"
  12. #include "BsUtil.h"
  13. #include "BsRenderWindowManager.h"
  14. #include "BsPlatform.h"
  15. #include "BsRectI.h"
  16. #include "BsCoreApplication.h"
  17. #include "BsException.h"
  18. #include "BsInput.h"
  19. #include "BsPass.h"
  20. #include "BsDebug.h"
  21. #include "BsDrawList.h"
  22. #include "BsGUIInputCaret.h"
  23. #include "BsGUIInputSelection.h"
  24. #include "BsGUIListBox.h"
  25. #include "BsGUIButton.h"
  26. #include "BsGUIDropDownBox.h"
  27. #include "BsGUIContextMenu.h"
  28. #include "BsDragAndDropManager.h"
  29. #include "BsGUIDropDownBoxManager.h"
  30. #include "BsGUIContextMenu.h"
  31. #include "BsProfilerCPU.h"
  32. #include "BsMeshHeap.h"
  33. #include "BsTransientMesh.h"
  34. #include "BsVirtualInput.h"
  35. #include "BsCursor.h"
  36. #include "BsCoreThread.h"
  37. using namespace std::placeholders;
  38. namespace BansheeEngine
  39. {
  40. struct GUIGroupElement
  41. {
  42. GUIGroupElement()
  43. { }
  44. GUIGroupElement(GUIElement* _element, UINT32 _renderElement)
  45. :element(_element), renderElement(_renderElement)
  46. { }
  47. GUIElement* element;
  48. UINT32 renderElement;
  49. };
  50. struct GUIMaterialGroup
  51. {
  52. GUIMaterialInfo matInfo;
  53. UINT32 numQuads;
  54. UINT32 depth;
  55. RectI bounds;
  56. Vector<GUIGroupElement> elements;
  57. };
  58. const UINT32 GUIManager::DRAG_DISTANCE = 3;
  59. const UINT32 GUIManager::MESH_HEAP_INITIAL_NUM_VERTS = 16384;
  60. const UINT32 GUIManager::MESH_HEAP_INITIAL_NUM_INDICES = 49152;
  61. GUIManager::GUIManager()
  62. :mSeparateMeshesByWidget(true), mActiveMouseButton(GUIMouseButton::Left),
  63. mCaretBlinkInterval(0.5f), mCaretLastBlinkTime(0.0f), mCaretColor(1.0f, 0.6588f, 0.0f), mIsCaretOn(false),
  64. mTextSelectionColor(1.0f, 0.6588f, 0.0f), mInputCaret(nullptr), mInputSelection(nullptr), mDragState(DragState::NoDrag),
  65. mActiveCursor(CursorType::Arrow)
  66. {
  67. mOnPointerMovedConn = gInput().onPointerMoved.connect(std::bind(&GUIManager::onPointerMoved, this, _1));
  68. mOnPointerPressedConn = gInput().onPointerPressed.connect(std::bind(&GUIManager::onPointerPressed, this, _1));
  69. mOnPointerReleasedConn = gInput().onPointerReleased.connect(std::bind(&GUIManager::onPointerReleased, this, _1));
  70. mOnPointerDoubleClick = gInput().onPointerDoubleClick.connect(std::bind(&GUIManager::onPointerDoubleClick, this, _1));
  71. mOnTextInputConn = gInput().onCharInput.connect(std::bind(&GUIManager::onTextInput, this, _1));
  72. mOnInputCommandConn = gInput().onInputCommand.connect(std::bind(&GUIManager::onInputCommandEntered, this, _1));
  73. mOnVirtualButtonDown = VirtualInput::instance().onButtonDown.connect(std::bind(&GUIManager::onVirtualButtonDown, this, _1, _2));
  74. mWindowGainedFocusConn = RenderWindowManager::instance().onFocusGained.connect(std::bind(&GUIManager::onWindowFocusGained, this, _1));
  75. mWindowLostFocusConn = RenderWindowManager::instance().onFocusLost.connect(std::bind(&GUIManager::onWindowFocusLost, this, _1));
  76. mMouseLeftWindowConn = Platform::onMouseLeftWindow.connect(std::bind(&GUIManager::onMouseLeftWindow, this, _1));
  77. mInputCaret = bs_new<GUIInputCaret, PoolAlloc>();
  78. mInputSelection = bs_new<GUIInputSelection, PoolAlloc>();
  79. DragAndDropManager::startUp();
  80. mDragEndedConn = DragAndDropManager::instance().onDragEnded.connect(std::bind(&GUIManager::onMouseDragEnded, this, _1, _2));
  81. GUIDropDownBoxManager::startUp();
  82. mVertexDesc = bs_shared_ptr<VertexDataDesc>();
  83. mVertexDesc->addVertElem(VET_FLOAT2, VES_POSITION);
  84. mVertexDesc->addVertElem(VET_FLOAT2, VES_TEXCOORD);
  85. mMeshHeap = MeshHeap::create(MESH_HEAP_INITIAL_NUM_VERTS, MESH_HEAP_INITIAL_NUM_INDICES, mVertexDesc);
  86. // Need to defer this call because I want to make sure all managers are initialized first
  87. deferredCall(std::bind(&GUIManager::updateCaretTexture, this));
  88. deferredCall(std::bind(&GUIManager::updateTextSelectionTexture, this));
  89. }
  90. GUIManager::~GUIManager()
  91. {
  92. GUIDropDownBoxManager::shutDown();
  93. DragAndDropManager::shutDown();
  94. // Make a copy of widgets, since destroying them will remove them from mWidgets and
  95. // we can't iterate over an array thats getting modified
  96. Vector<WidgetInfo> widgetCopy = mWidgets;
  97. for(auto& widget : widgetCopy)
  98. widget.widget->destroy();
  99. // Ensure everything queued get destroyed
  100. processDestroyQueue();
  101. mOnPointerPressedConn.disconnect();
  102. mOnPointerReleasedConn.disconnect();
  103. mOnPointerMovedConn.disconnect();
  104. mOnPointerDoubleClick.disconnect();
  105. mOnTextInputConn.disconnect();
  106. mOnInputCommandConn.disconnect();
  107. mOnVirtualButtonDown.disconnect();
  108. mDragEndedConn.disconnect();
  109. mWindowGainedFocusConn.disconnect();
  110. mWindowLostFocusConn.disconnect();
  111. mMouseLeftWindowConn.disconnect();
  112. bs_delete<PoolAlloc>(mInputCaret);
  113. bs_delete<PoolAlloc>(mInputSelection);
  114. }
  115. void GUIManager::registerWidget(GUIWidget* widget)
  116. {
  117. mWidgets.push_back(WidgetInfo(widget));
  118. const Viewport* renderTarget = widget->getTarget();
  119. auto findIter = mCachedGUIData.find(renderTarget);
  120. if(findIter == end(mCachedGUIData))
  121. mCachedGUIData[renderTarget] = GUIRenderData();
  122. GUIRenderData& windowData = mCachedGUIData[renderTarget];
  123. windowData.widgets.push_back(widget);
  124. windowData.isDirty = true;
  125. }
  126. void GUIManager::unregisterWidget(GUIWidget* widget)
  127. {
  128. {
  129. auto findIter = std::find_if(begin(mWidgets), end(mWidgets), [=] (const WidgetInfo& x) { return x.widget == widget; } );
  130. if(findIter != mWidgets.end())
  131. mWidgets.erase(findIter);
  132. }
  133. const Viewport* renderTarget = widget->getTarget();
  134. GUIRenderData& renderData = mCachedGUIData[renderTarget];
  135. {
  136. auto findIter = std::find(begin(renderData.widgets), end(renderData.widgets), widget);
  137. if(findIter != end(renderData.widgets))
  138. renderData.widgets.erase(findIter);
  139. }
  140. if(renderData.widgets.size() == 0)
  141. {
  142. mCachedGUIData.erase(renderTarget);
  143. }
  144. else
  145. renderData.isDirty = true;
  146. }
  147. void GUIManager::update()
  148. {
  149. DragAndDropManager::instance()._update();
  150. // Update layouts
  151. gProfilerCPU().beginSample("UpdateLayout");
  152. for(auto& widgetInfo : mWidgets)
  153. {
  154. widgetInfo.widget->_updateLayout();
  155. }
  156. gProfilerCPU().endSample("UpdateLayout");
  157. // Blink caret
  158. float curTime = gTime().getTime();
  159. if((curTime - mCaretLastBlinkTime) >= mCaretBlinkInterval)
  160. {
  161. mCaretLastBlinkTime = curTime;
  162. mIsCaretOn = !mIsCaretOn;
  163. mCommandEvent = GUICommandEvent();
  164. mCommandEvent.setType(GUICommandEventType::Redraw);
  165. for(auto& elementInfo : mElementsInFocus)
  166. {
  167. sendCommandEvent(elementInfo.widget, elementInfo.element, mCommandEvent);
  168. }
  169. }
  170. PROFILE_CALL(updateMeshes(), "UpdateMeshes");
  171. mNewElementsUnderPointer.clear();
  172. for(auto& elementInfo : mElementsUnderPointer)
  173. {
  174. if(!elementInfo.element->_isDestroyed())
  175. mNewElementsUnderPointer.push_back(elementInfo);
  176. }
  177. mElementsUnderPointer.swap(mNewElementsUnderPointer);
  178. mNewActiveElements.clear();
  179. for(auto& elementInfo : mActiveElements)
  180. {
  181. if(!elementInfo.element->_isDestroyed())
  182. mNewActiveElements.push_back(elementInfo);
  183. }
  184. mActiveElements.swap(mNewActiveElements);
  185. mNewElementsInFocus.clear();
  186. for(auto& elementInfo : mElementsInFocus)
  187. {
  188. if(!elementInfo.element->_isDestroyed())
  189. mNewElementsInFocus.push_back(elementInfo);
  190. }
  191. mElementsInFocus.swap(mNewElementsInFocus);
  192. for(auto& focusElementInfo : mForcedFocusElements)
  193. {
  194. if(focusElementInfo.element->_isDestroyed())
  195. continue;
  196. if(focusElementInfo.focus)
  197. {
  198. auto iterFind = std::find_if(mElementsInFocus.begin(), mElementsInFocus.end(),
  199. [&](const ElementInfo& x) { return x.element == focusElementInfo.element; });
  200. if(iterFind == mElementsInFocus.end())
  201. {
  202. mElementsInFocus.push_back(ElementInfo(focusElementInfo.element, focusElementInfo.element->_getParentWidget()));
  203. mCommandEvent = GUICommandEvent();
  204. mCommandEvent.setType(GUICommandEventType::FocusGained);
  205. sendCommandEvent(focusElementInfo.element->_getParentWidget(), focusElementInfo.element, mCommandEvent);
  206. }
  207. }
  208. else
  209. {
  210. mNewElementsInFocus.clear();
  211. for(auto& elementInfo : mElementsInFocus)
  212. {
  213. if(elementInfo.element == focusElementInfo.element)
  214. {
  215. mCommandEvent = GUICommandEvent();
  216. mCommandEvent.setType(GUICommandEventType::FocusLost);
  217. sendCommandEvent(elementInfo.widget, elementInfo.element, mCommandEvent);
  218. }
  219. else
  220. mNewElementsInFocus.push_back(elementInfo);
  221. }
  222. mElementsInFocus.swap(mNewElementsInFocus);
  223. }
  224. }
  225. mForcedFocusElements.clear();
  226. processDestroyQueue();
  227. }
  228. void GUIManager::render(ViewportPtr& target, DrawList& drawList) const
  229. {
  230. auto findIter = mCachedGUIData.find(target.get());
  231. if(findIter == mCachedGUIData.end())
  232. return;
  233. const GUIRenderData& renderData = findIter->second;
  234. // Render the meshes
  235. if(mSeparateMeshesByWidget)
  236. {
  237. // TODO - Possible optimization. I currently divide by width/height inside the shader, while it
  238. // might be more optimal to just scale the mesh as the resolution changes?
  239. float invViewportWidth = 1.0f / (target->getWidth() * 0.5f);
  240. float invViewportHeight = 1.0f / (target->getHeight() * 0.5f);
  241. UINT32 meshIdx = 0;
  242. for(auto& mesh : renderData.cachedMeshes)
  243. {
  244. GUIMaterialInfo materialInfo = renderData.cachedMaterials[meshIdx];
  245. GUIWidget* widget = renderData.cachedWidgetsPerMesh[meshIdx];
  246. if(materialInfo.material == nullptr || !materialInfo.material.isLoaded())
  247. {
  248. meshIdx++;
  249. continue;
  250. }
  251. if(mesh == nullptr)
  252. {
  253. meshIdx++;
  254. continue;
  255. }
  256. materialInfo.invViewportWidth.set(invViewportWidth);
  257. materialInfo.invViewportHeight.set(invViewportHeight);
  258. materialInfo.worldTransform.set(widget->SO()->getWorldTfrm());
  259. drawList.add(materialInfo.material.getInternalPtr(), mesh, 0, Vector3::ZERO);
  260. meshIdx++;
  261. }
  262. }
  263. else
  264. {
  265. // TODO: I want to avoid separating meshes by widget in the future. On DX11 and GL I can set up a shader
  266. // that accepts multiple world transforms (one for each widget). Then I can add some instance information to vertices
  267. // and render elements using multiple different transforms with a single call.
  268. // Separating meshes can then be used as a compatibility mode for DX9
  269. BS_EXCEPT(NotImplementedException, "Not implemented");
  270. }
  271. }
  272. void GUIManager::updateMeshes()
  273. {
  274. for(auto& cachedMeshData : mCachedGUIData)
  275. {
  276. GUIRenderData& renderData = cachedMeshData.second;
  277. // Check if anything is dirty. If nothing is we can skip the update
  278. bool isDirty = renderData.isDirty;
  279. renderData.isDirty = false;
  280. for(auto& widget : renderData.widgets)
  281. {
  282. if(widget->isDirty(true))
  283. {
  284. isDirty = true;
  285. }
  286. }
  287. if(!isDirty)
  288. continue;
  289. // Make a list of all GUI elements, sorted from farthest to nearest (highest depth to lowest)
  290. auto elemComp = [](const GUIGroupElement& a, const GUIGroupElement& b)
  291. {
  292. UINT32 aDepth = a.element->_getRenderElementDepth(a.renderElement);
  293. UINT32 bDepth = b.element->_getRenderElementDepth(b.renderElement);
  294. // Compare pointers just to differentiate between two elements with the same depth, their order doesn't really matter, but std::set
  295. // requires all elements to be unique
  296. return (aDepth > bDepth) ||
  297. (aDepth == bDepth && a.element > b.element) ||
  298. (aDepth == bDepth && a.element == b.element && a.renderElement > b.renderElement);
  299. };
  300. Set<GUIGroupElement, std::function<bool(const GUIGroupElement&, const GUIGroupElement&)>> allElements(elemComp);
  301. for(auto& widget : renderData.widgets)
  302. {
  303. const Vector<GUIElement*>& elements = widget->getElements();
  304. for(auto& element : elements)
  305. {
  306. if(element->_isDisabled())
  307. continue;
  308. UINT32 numRenderElems = element->getNumRenderElements();
  309. for(UINT32 i = 0; i < numRenderElems; i++)
  310. {
  311. allElements.insert(GUIGroupElement(element, i));
  312. }
  313. }
  314. }
  315. // Group the elements in such a way so that we end up with a smallest amount of
  316. // meshes, without breaking back to front rendering order
  317. UnorderedMap<UINT64, Vector<GUIMaterialGroup>> materialGroups;
  318. for(auto& elem : allElements)
  319. {
  320. GUIElement* guiElem = elem.element;
  321. UINT32 renderElemIdx = elem.renderElement;
  322. UINT32 elemDepth = guiElem->_getRenderElementDepth(renderElemIdx);
  323. RectI tfrmedBounds = guiElem->_getClippedBounds();
  324. tfrmedBounds.transform(guiElem->_getParentWidget()->SO()->getWorldTfrm());
  325. const GUIMaterialInfo& matInfo = guiElem->getMaterial(renderElemIdx);
  326. UINT64 materialId = matInfo.material->getInternalID(); // TODO - I group based on material ID. So if two widgets used exact copies of the same material
  327. // this system won't detect it. Find a better way of determining material similarity?
  328. // If this is a new material, add a new list of groups
  329. auto findIterMaterial = materialGroups.find(materialId);
  330. if(findIterMaterial == end(materialGroups))
  331. materialGroups[materialId] = Vector<GUIMaterialGroup>();
  332. // Try to find a group this material will fit in:
  333. // - Group that has a depth value same or one below elements depth will always be a match
  334. // - Otherwise, we search higher depth values as well, but we only use them if no elements in between those depth values
  335. // overlap the current elements bounds.
  336. Vector<GUIMaterialGroup>& allGroups = materialGroups[materialId];
  337. GUIMaterialGroup* foundGroup = nullptr;
  338. for(auto groupIter = allGroups.rbegin(); groupIter != allGroups.rend(); ++groupIter)
  339. {
  340. // If we separate meshes by widget, ignore any groups with widget parents other than mine
  341. if(mSeparateMeshesByWidget)
  342. {
  343. if(groupIter->elements.size() > 0)
  344. {
  345. GUIElement* otherElem = groupIter->elements.begin()->element; // We only need to check the first element
  346. if(otherElem->_getParentWidget() != guiElem->_getParentWidget())
  347. continue;
  348. }
  349. }
  350. GUIMaterialGroup& group = *groupIter;
  351. if(group.depth == elemDepth || group.depth == (elemDepth - 1))
  352. {
  353. foundGroup = &group;
  354. break;
  355. }
  356. else
  357. {
  358. UINT32 startDepth = elemDepth;
  359. UINT32 endDepth = group.depth;
  360. RectI potentialGroupBounds = group.bounds;
  361. potentialGroupBounds.encapsulate(tfrmedBounds);
  362. bool foundOverlap = false;
  363. for(auto& material : materialGroups)
  364. {
  365. for(auto& matGroup : material.second)
  366. {
  367. if(&matGroup == &group)
  368. continue;
  369. if(matGroup.depth > startDepth && matGroup.depth < endDepth)
  370. {
  371. if(matGroup.bounds.overlaps(potentialGroupBounds))
  372. {
  373. foundOverlap = true;
  374. break;
  375. }
  376. }
  377. }
  378. }
  379. if(!foundOverlap)
  380. {
  381. foundGroup = &group;
  382. break;
  383. }
  384. }
  385. }
  386. if(foundGroup == nullptr)
  387. {
  388. allGroups.push_back(GUIMaterialGroup());
  389. foundGroup = &allGroups[allGroups.size() - 1];
  390. foundGroup->depth = elemDepth;
  391. foundGroup->bounds = tfrmedBounds;
  392. foundGroup->elements.push_back(GUIGroupElement(guiElem, renderElemIdx));
  393. foundGroup->matInfo = matInfo;
  394. foundGroup->numQuads = guiElem->getNumQuads(renderElemIdx);
  395. }
  396. else
  397. {
  398. foundGroup->bounds.encapsulate(tfrmedBounds);
  399. foundGroup->elements.push_back(GUIGroupElement(guiElem, renderElemIdx));
  400. foundGroup->depth = std::min(foundGroup->depth, elemDepth);
  401. foundGroup->numQuads += guiElem->getNumQuads(renderElemIdx);
  402. }
  403. }
  404. // Make a list of all GUI elements, sorted from farthest to nearest (highest depth to lowest)
  405. auto groupComp = [](GUIMaterialGroup* a, GUIMaterialGroup* b)
  406. {
  407. return (a->depth > b->depth) || (a->depth == b->depth && a > b);
  408. // Compare pointers just to differentiate between two elements with the same depth, their order doesn't really matter, but std::set
  409. // requires all elements to be unique
  410. };
  411. Set<GUIMaterialGroup*, std::function<bool(GUIMaterialGroup*, GUIMaterialGroup*)>> sortedGroups(groupComp);
  412. for(auto& material : materialGroups)
  413. {
  414. for(auto& group : material.second)
  415. {
  416. sortedGroups.insert(&group);
  417. }
  418. }
  419. UINT32 numMeshes = (UINT32)sortedGroups.size();
  420. UINT32 oldNumMeshes = (UINT32)renderData.cachedMeshes.size();
  421. if(numMeshes < oldNumMeshes)
  422. {
  423. renderData.cachedMeshes.resize(numMeshes);
  424. }
  425. renderData.cachedMaterials.resize(numMeshes);
  426. if(mSeparateMeshesByWidget)
  427. renderData.cachedWidgetsPerMesh.resize(numMeshes);
  428. // Fill buffers for each group and update their meshes
  429. UINT32 groupIdx = 0;
  430. for(auto& group : sortedGroups)
  431. {
  432. renderData.cachedMaterials[groupIdx] = group->matInfo;
  433. if(mSeparateMeshesByWidget)
  434. {
  435. if(group->elements.size() == 0)
  436. renderData.cachedWidgetsPerMesh[groupIdx] = nullptr;
  437. else
  438. {
  439. GUIElement* elem = group->elements.begin()->element;
  440. renderData.cachedWidgetsPerMesh[groupIdx] = elem->_getParentWidget();
  441. }
  442. }
  443. MeshDataPtr meshData = bs_shared_ptr<MeshData, PoolAlloc>(group->numQuads * 4, group->numQuads * 6, mVertexDesc);
  444. UINT8* vertices = meshData->getElementData(VES_POSITION);
  445. UINT8* uvs = meshData->getElementData(VES_TEXCOORD);
  446. UINT32* indices = meshData->getIndices32();
  447. UINT32 vertexStride = meshData->getVertexDesc()->getVertexStride();
  448. UINT32 indexStride = meshData->getIndexElementSize();
  449. UINT32 quadOffset = 0;
  450. for(auto& matElement : group->elements)
  451. {
  452. matElement.element->fillBuffer(vertices, uvs, indices, quadOffset, group->numQuads, vertexStride, indexStride, matElement.renderElement);
  453. UINT32 numQuads = matElement.element->getNumQuads(matElement.renderElement);
  454. UINT32 indexStart = quadOffset * 6;
  455. UINT32 indexEnd = indexStart + numQuads * 6;
  456. UINT32 vertOffset = quadOffset * 4;
  457. for(UINT32 i = indexStart; i < indexEnd; i++)
  458. indices[i] += vertOffset;
  459. quadOffset += numQuads;
  460. }
  461. if(groupIdx < (UINT32)renderData.cachedMeshes.size())
  462. {
  463. mMeshHeap->dealloc(renderData.cachedMeshes[groupIdx]);
  464. renderData.cachedMeshes[groupIdx] = mMeshHeap->alloc(meshData);
  465. }
  466. else
  467. {
  468. renderData.cachedMeshes.push_back(mMeshHeap->alloc(meshData));
  469. }
  470. groupIdx++;
  471. }
  472. }
  473. }
  474. void GUIManager::updateCaretTexture()
  475. {
  476. if(mCaretTexture == nullptr)
  477. {
  478. HTexture newTex = Texture::create(TEX_TYPE_2D, 1, 1, 0, PF_R8G8B8A8);
  479. newTex->synchronize(); // TODO - Required due to a bug in allocateSubresourceBuffer
  480. mCaretTexture = SpriteTexture::create(newTex);
  481. }
  482. const HTexture& tex = mCaretTexture->getTexture();
  483. UINT32 subresourceIdx = tex->mapToSubresourceIdx(0, 0);
  484. PixelDataPtr data = tex->allocateSubresourceBuffer(subresourceIdx);
  485. data->setColorAt(mCaretColor, 0, 0);
  486. gCoreAccessor().writeSubresource(tex.getInternalPtr(), tex->mapToSubresourceIdx(0, 0), data);
  487. }
  488. void GUIManager::updateTextSelectionTexture()
  489. {
  490. if(mTextSelectionTexture == nullptr)
  491. {
  492. HTexture newTex = Texture::create(TEX_TYPE_2D, 1, 1, 0, PF_R8G8B8A8);
  493. newTex->synchronize(); // TODO - Required due to a bug in allocateSubresourceBuffer
  494. mTextSelectionTexture = SpriteTexture::create(newTex);
  495. }
  496. const HTexture& tex = mTextSelectionTexture->getTexture();
  497. UINT32 subresourceIdx = tex->mapToSubresourceIdx(0, 0);
  498. PixelDataPtr data = tex->allocateSubresourceBuffer(subresourceIdx);
  499. data->setColorAt(mTextSelectionColor, 0, 0);
  500. gCoreAccessor().writeSubresource(tex.getInternalPtr(), tex->mapToSubresourceIdx(0, 0), data);
  501. }
  502. void GUIManager::onMouseDragEnded(const PointerEvent& event, DragCallbackInfo& dragInfo)
  503. {
  504. GUIMouseButton guiButton = buttonToGUIButton(event.button);
  505. if(DragAndDropManager::instance().isDragInProgress() && guiButton == GUIMouseButton::Left)
  506. {
  507. for(auto& elementInfo : mElementsUnderPointer)
  508. {
  509. Vector2I localPos;
  510. if(elementInfo.widget != nullptr)
  511. localPos = getWidgetRelativePos(*elementInfo.widget, event.screenPos);
  512. bool acceptDrop = true;
  513. if(DragAndDropManager::instance().needsValidDropTarget())
  514. {
  515. acceptDrop = elementInfo.element->_acceptDragAndDrop(localPos, DragAndDropManager::instance().getDragTypeId());
  516. }
  517. if(acceptDrop)
  518. {
  519. mMouseEvent.setDragAndDropDroppedData(localPos, DragAndDropManager::instance().getDragTypeId(), DragAndDropManager::instance().getDragData());
  520. dragInfo.processed = sendMouseEvent(elementInfo.widget, elementInfo.element, mMouseEvent);
  521. if(dragInfo.processed)
  522. return;
  523. }
  524. }
  525. }
  526. dragInfo.processed = false;
  527. }
  528. void GUIManager::onPointerMoved(const PointerEvent& event)
  529. {
  530. if(event.isUsed())
  531. return;
  532. bool buttonStates[(int)GUIMouseButton::Count];
  533. buttonStates[0] = event.buttonStates[0];
  534. buttonStates[1] = event.buttonStates[1];
  535. buttonStates[2] = event.buttonStates[2];
  536. if(findElementUnderPointer(event.screenPos, buttonStates, event.shift, event.control, event.alt))
  537. event.markAsUsed();
  538. if(mDragState == DragState::HeldWithoutDrag)
  539. {
  540. UINT32 dist = mLastPointerClickPos.manhattanDist(event.screenPos);
  541. if(dist > DRAG_DISTANCE)
  542. {
  543. for(auto& activeElement : mActiveElements)
  544. {
  545. Vector2I localPos = getWidgetRelativePos(*activeElement.widget, event.screenPos);
  546. mMouseEvent.setMouseDragStartData(localPos);
  547. if(sendMouseEvent(activeElement.widget, activeElement.element, mMouseEvent))
  548. event.markAsUsed();
  549. }
  550. mDragState = DragState::Dragging;
  551. }
  552. }
  553. // If mouse is being held down send MouseDrag events
  554. if(mDragState == DragState::Dragging)
  555. {
  556. for(auto& activeElement : mActiveElements)
  557. {
  558. if(mLastPointerScreenPos != event.screenPos)
  559. {
  560. Vector2I localPos = getWidgetRelativePos(*activeElement.widget, event.screenPos);
  561. mMouseEvent.setMouseDragData(localPos, localPos - mLastPointerScreenPos);
  562. if(sendMouseEvent(activeElement.widget, activeElement.element, mMouseEvent))
  563. event.markAsUsed();
  564. }
  565. }
  566. mLastPointerScreenPos = event.screenPos;
  567. // Also if drag is in progress send DragAndDrop events
  568. if(DragAndDropManager::instance().isDragInProgress())
  569. {
  570. bool acceptDrop = true;
  571. for(auto& elementInfo : mElementsUnderPointer)
  572. {
  573. Vector2I localPos = getWidgetRelativePos(*elementInfo.widget, event.screenPos);
  574. acceptDrop = true;
  575. if(DragAndDropManager::instance().needsValidDropTarget())
  576. {
  577. acceptDrop = elementInfo.element->_acceptDragAndDrop(localPos, DragAndDropManager::instance().getDragTypeId());
  578. }
  579. if(acceptDrop)
  580. {
  581. mMouseEvent.setDragAndDropDraggedData(localPos, DragAndDropManager::instance().getDragTypeId(), DragAndDropManager::instance().getDragData());
  582. if(sendMouseEvent(elementInfo.widget, elementInfo.element, mMouseEvent))
  583. {
  584. event.markAsUsed();
  585. break;
  586. }
  587. }
  588. }
  589. if(acceptDrop)
  590. {
  591. if(mActiveCursor != CursorType::ArrowDrag)
  592. {
  593. Cursor::instance().setCursor(CursorType::ArrowDrag);
  594. mActiveCursor = CursorType::ArrowDrag;
  595. }
  596. }
  597. else
  598. {
  599. if(mActiveCursor != CursorType::Deny)
  600. {
  601. Cursor::instance().setCursor(CursorType::Deny);
  602. mActiveCursor = CursorType::Deny;
  603. }
  604. }
  605. }
  606. }
  607. else // Otherwise, send MouseMove events if we are hovering over any element
  608. {
  609. if(mLastPointerScreenPos != event.screenPos)
  610. {
  611. bool moveProcessed = false;
  612. bool hasCustomCursor = false;
  613. for(auto& elementInfo : mElementsUnderPointer)
  614. {
  615. Vector2I localPos = getWidgetRelativePos(*elementInfo.widget, event.screenPos);
  616. if(!moveProcessed)
  617. {
  618. // Send MouseMove event
  619. mMouseEvent.setMouseMoveData(localPos);
  620. moveProcessed = sendMouseEvent(elementInfo.widget, elementInfo.element, mMouseEvent);
  621. if(moveProcessed)
  622. {
  623. event.markAsUsed();
  624. break;
  625. }
  626. }
  627. if(!hasCustomCursor)
  628. {
  629. CursorType newCursor = CursorType::Arrow;
  630. if(elementInfo.element->_hasCustomCursor(localPos, newCursor))
  631. {
  632. if(newCursor != mActiveCursor)
  633. {
  634. Cursor::instance().setCursor(newCursor);
  635. mActiveCursor = newCursor;
  636. }
  637. hasCustomCursor = true;
  638. }
  639. }
  640. if(moveProcessed && hasCustomCursor)
  641. break;
  642. }
  643. if(!hasCustomCursor)
  644. {
  645. if(mActiveCursor != CursorType::Arrow)
  646. {
  647. Cursor::instance().setCursor(CursorType::Arrow);
  648. mActiveCursor = CursorType::Arrow;
  649. }
  650. }
  651. }
  652. mLastPointerScreenPos = event.screenPos;
  653. if(Math::abs(event.mouseWheelScrollAmount) > 0.00001f)
  654. {
  655. for(auto& elementInfo : mElementsUnderPointer)
  656. {
  657. mMouseEvent.setMouseWheelScrollData(event.mouseWheelScrollAmount);
  658. if(sendMouseEvent(elementInfo.widget, elementInfo.element, mMouseEvent))
  659. {
  660. event.markAsUsed();
  661. break;
  662. }
  663. }
  664. }
  665. }
  666. }
  667. void GUIManager::onPointerReleased(const PointerEvent& event)
  668. {
  669. if(event.isUsed())
  670. return;
  671. bool buttonStates[(int)GUIMouseButton::Count];
  672. buttonStates[0] = event.buttonStates[0];
  673. buttonStates[1] = event.buttonStates[1];
  674. buttonStates[2] = event.buttonStates[2];
  675. if(findElementUnderPointer(event.screenPos, buttonStates, event.shift, event.control, event.alt))
  676. event.markAsUsed();
  677. mMouseEvent = GUIMouseEvent(buttonStates, event.shift, event.control, event.alt);
  678. GUIMouseButton guiButton = buttonToGUIButton(event.button);
  679. // Send MouseUp event only if we are over the active element (we don't want to accidentally trigger other elements).
  680. // And only activate when a button that originally caused the active state is released, otherwise ignore it.
  681. if(mActiveMouseButton == guiButton)
  682. {
  683. for(auto& elementInfo : mElementsUnderPointer)
  684. {
  685. auto iterFind2 = std::find_if(mActiveElements.begin(), mActiveElements.end(),
  686. [&](const ElementInfo& x) { return x.element == elementInfo.element; });
  687. if(iterFind2 != mActiveElements.end())
  688. {
  689. Vector2I localPos = getWidgetRelativePos(*elementInfo.widget, event.screenPos);
  690. mMouseEvent.setMouseUpData(localPos, guiButton);
  691. if(sendMouseEvent(elementInfo.widget, elementInfo.element, mMouseEvent))
  692. {
  693. event.markAsUsed();
  694. break;
  695. }
  696. }
  697. }
  698. }
  699. // Send DragEnd event to whichever element is active
  700. bool acceptEndDrag = (mDragState == DragState::Dragging || mDragState == DragState::HeldWithoutDrag) && mActiveMouseButton == guiButton &&
  701. (guiButton == GUIMouseButton::Left);
  702. if(acceptEndDrag)
  703. {
  704. if(mDragState == DragState::Dragging)
  705. {
  706. for(auto& activeElement : mActiveElements)
  707. {
  708. Vector2I localPos = getWidgetRelativePos(*activeElement.widget, event.screenPos);
  709. mMouseEvent.setMouseDragEndData(localPos);
  710. if(sendMouseEvent(activeElement.widget, activeElement.element, mMouseEvent))
  711. event.markAsUsed();
  712. }
  713. }
  714. mDragState = DragState::NoDrag;
  715. }
  716. if(mActiveMouseButton == guiButton)
  717. {
  718. mActiveElements.clear();
  719. mActiveMouseButton = GUIMouseButton::Left;
  720. }
  721. if(mActiveCursor != CursorType::Arrow)
  722. {
  723. Cursor::instance().setCursor(CursorType::Arrow);
  724. mActiveCursor = CursorType::Arrow;
  725. }
  726. }
  727. void GUIManager::onPointerPressed(const PointerEvent& event)
  728. {
  729. if(event.isUsed())
  730. return;
  731. bool buttonStates[(int)GUIMouseButton::Count];
  732. buttonStates[0] = event.buttonStates[0];
  733. buttonStates[1] = event.buttonStates[1];
  734. buttonStates[2] = event.buttonStates[2];
  735. if(findElementUnderPointer(event.screenPos, buttonStates, event.shift, event.control, event.alt))
  736. event.markAsUsed();
  737. mMouseEvent = GUIMouseEvent(buttonStates, event.shift, event.control, event.alt);
  738. GUIMouseButton guiButton = buttonToGUIButton(event.button);
  739. // We only check for mouse down if mouse isn't already being held down, and we are hovering over an element
  740. if(mActiveElements.size() == 0)
  741. {
  742. mNewActiveElements.clear();
  743. for(auto& elementInfo : mElementsUnderPointer)
  744. {
  745. Vector2I localPos = getWidgetRelativePos(*elementInfo.widget, event.screenPos);
  746. mMouseEvent.setMouseDownData(localPos, guiButton);
  747. bool processed = sendMouseEvent(elementInfo.widget, elementInfo.element, mMouseEvent);
  748. if(guiButton == GUIMouseButton::Left)
  749. {
  750. mDragState = DragState::HeldWithoutDrag;
  751. mLastPointerClickPos = event.screenPos;
  752. }
  753. mNewActiveElements.push_back(ElementInfo(elementInfo.element, elementInfo.widget));
  754. mActiveMouseButton = guiButton;
  755. if(processed)
  756. {
  757. event.markAsUsed();
  758. break;
  759. }
  760. }
  761. mActiveElements.swap(mNewActiveElements);
  762. }
  763. mNewElementsInFocus.clear();
  764. mCommandEvent = GUICommandEvent();
  765. // Determine elements that gained focus
  766. mCommandEvent.setType(GUICommandEventType::FocusGained);
  767. for(auto& elementInfo : mElementsUnderPointer)
  768. {
  769. mNewElementsInFocus.push_back(elementInfo);
  770. auto iterFind = std::find_if(begin(mElementsInFocus), end(mElementsInFocus),
  771. [=] (const ElementInfo& x) { return x.element == elementInfo.element; });
  772. if(iterFind == mElementsInFocus.end())
  773. {
  774. sendCommandEvent(elementInfo.widget, elementInfo.element, mCommandEvent);
  775. }
  776. }
  777. // Determine elements that lost focus
  778. mCommandEvent.setType(GUICommandEventType::FocusLost);
  779. for(auto& elementInfo : mElementsInFocus)
  780. {
  781. auto iterFind = std::find_if(begin(mNewElementsInFocus), end(mNewElementsInFocus),
  782. [=] (const ElementInfo& x) { return x.element == elementInfo.element; });
  783. if(iterFind == mNewElementsInFocus.end())
  784. {
  785. sendCommandEvent(elementInfo.widget, elementInfo.element, mCommandEvent);
  786. }
  787. }
  788. if(mElementsUnderPointer.size() > 0)
  789. event.markAsUsed();
  790. mElementsInFocus.swap(mNewElementsInFocus);
  791. // If right click try to open context menu
  792. if(buttonStates[2] == true)
  793. {
  794. for(auto& elementInfo : mElementsUnderPointer)
  795. {
  796. GUIContextMenu* menu = elementInfo.element->getContextMenu();
  797. if(menu != nullptr)
  798. {
  799. const RenderWindow* window = getWidgetWindow(*elementInfo.widget);
  800. Vector2I windowPos = window->screenToWindowPos(event.screenPos);
  801. menu->open(windowPos, *elementInfo.widget);
  802. event.markAsUsed();
  803. break;
  804. }
  805. }
  806. }
  807. }
  808. void GUIManager::onPointerDoubleClick(const PointerEvent& event)
  809. {
  810. if(event.isUsed())
  811. return;
  812. bool buttonStates[(int)GUIMouseButton::Count];
  813. buttonStates[0] = event.buttonStates[0];
  814. buttonStates[1] = event.buttonStates[1];
  815. buttonStates[2] = event.buttonStates[2];
  816. if(findElementUnderPointer(event.screenPos, buttonStates, event.shift, event.control, event.alt))
  817. event.markAsUsed();
  818. mMouseEvent = GUIMouseEvent(buttonStates, event.shift, event.control, event.alt);
  819. GUIMouseButton guiButton = buttonToGUIButton(event.button);
  820. // We only check for mouse down if we are hovering over an element
  821. for(auto& elementInfo : mElementsUnderPointer)
  822. {
  823. Vector2I localPos = getWidgetRelativePos(*elementInfo.widget, event.screenPos);
  824. mMouseEvent.setMouseDoubleClickData(localPos, guiButton);
  825. if(sendMouseEvent(elementInfo.widget, elementInfo.element, mMouseEvent))
  826. {
  827. event.markAsUsed();
  828. break;
  829. }
  830. }
  831. }
  832. void GUIManager::onInputCommandEntered(InputCommandType commandType)
  833. {
  834. if(mElementsInFocus.size() == 0)
  835. return;
  836. mCommandEvent = GUICommandEvent();
  837. switch(commandType)
  838. {
  839. case InputCommandType::Backspace:
  840. mCommandEvent.setType(GUICommandEventType::Backspace);
  841. break;
  842. case InputCommandType::Delete:
  843. mCommandEvent.setType(GUICommandEventType::Delete);
  844. break;
  845. case InputCommandType::Return:
  846. mCommandEvent.setType(GUICommandEventType::Return);
  847. break;
  848. case InputCommandType::Escape:
  849. mCommandEvent.setType(GUICommandEventType::Escape);
  850. break;
  851. case InputCommandType::CursorMoveLeft:
  852. mCommandEvent.setType(GUICommandEventType::MoveLeft);
  853. break;
  854. case InputCommandType::CursorMoveRight:
  855. mCommandEvent.setType(GUICommandEventType::MoveRight);
  856. break;
  857. case InputCommandType::CursorMoveUp:
  858. mCommandEvent.setType(GUICommandEventType::MoveUp);
  859. break;
  860. case InputCommandType::CursorMoveDown:
  861. mCommandEvent.setType(GUICommandEventType::MoveDown);
  862. break;
  863. case InputCommandType::SelectLeft:
  864. mCommandEvent.setType(GUICommandEventType::SelectLeft);
  865. break;
  866. case InputCommandType::SelectRight:
  867. mCommandEvent.setType(GUICommandEventType::SelectRight);
  868. break;
  869. case InputCommandType::SelectUp:
  870. mCommandEvent.setType(GUICommandEventType::SelectUp);
  871. break;
  872. case InputCommandType::SelectDown:
  873. mCommandEvent.setType(GUICommandEventType::SelectDown);
  874. break;
  875. }
  876. for(auto& elementInfo : mElementsInFocus)
  877. {
  878. sendCommandEvent(elementInfo.widget, elementInfo.element, mCommandEvent);
  879. }
  880. }
  881. void GUIManager::onVirtualButtonDown(const VirtualButton& button, UINT32 deviceIdx)
  882. {
  883. mVirtualButtonEvent.setButton(button);
  884. for(auto& elementInFocus : mElementsInFocus)
  885. {
  886. bool processed = sendVirtualButtonEvent(elementInFocus.widget, elementInFocus.element, mVirtualButtonEvent);
  887. if(processed)
  888. break;
  889. }
  890. }
  891. bool GUIManager::findElementUnderPointer(const Vector2I& pointerScreenPos, bool buttonStates[3], bool shift, bool control, bool alt)
  892. {
  893. Vector<const RenderWindow*> widgetWindows;
  894. for(auto& widgetInfo : mWidgets)
  895. widgetWindows.push_back(getWidgetWindow(*widgetInfo.widget));
  896. #if BS_DEBUG_MODE
  897. // Checks if all referenced windows actually exist
  898. Vector<RenderWindow*> activeWindows = RenderWindowManager::instance().getRenderWindows();
  899. for(auto& window : widgetWindows)
  900. {
  901. if(window == nullptr)
  902. continue;
  903. auto iterFind = std::find(begin(activeWindows), end(activeWindows), window);
  904. if(iterFind == activeWindows.end())
  905. {
  906. BS_EXCEPT(InternalErrorException, "GUI manager has a reference to a window that doesn't exist. \
  907. Please detach all GUIWidgets from windows before destroying a window.");
  908. }
  909. }
  910. #endif
  911. mNewElementsUnderPointer.clear();
  912. const RenderWindow* windowUnderPointer = nullptr;
  913. UnorderedSet<const RenderWindow*> uniqueWindows;
  914. for(auto& window : widgetWindows)
  915. {
  916. if(window == nullptr)
  917. continue;
  918. uniqueWindows.insert(window);
  919. }
  920. for(auto& window : uniqueWindows)
  921. {
  922. if(Platform::isPointOverWindow(*window, pointerScreenPos))
  923. {
  924. windowUnderPointer = window;
  925. break;
  926. }
  927. }
  928. if(windowUnderPointer != nullptr)
  929. {
  930. Vector2I windowPos = windowUnderPointer->screenToWindowPos(pointerScreenPos);
  931. Vector4 vecWindowPos((float)windowPos.x, (float)windowPos.y, 0.0f, 1.0f);
  932. UINT32 widgetIdx = 0;
  933. for(auto& widgetInfo : mWidgets)
  934. {
  935. if(widgetWindows[widgetIdx] == nullptr)
  936. {
  937. widgetIdx++;
  938. continue;
  939. }
  940. GUIWidget* widget = widgetInfo.widget;
  941. if(widgetWindows[widgetIdx] == windowUnderPointer && widget->inBounds(windowToBridgedCoords(*widget, windowPos)))
  942. {
  943. const Vector<GUIElement*>& elements = widget->getElements();
  944. Vector2I localPos = getWidgetRelativePos(*widget, pointerScreenPos);
  945. // Elements with lowest depth (most to the front) get handled first
  946. for(auto iter = elements.begin(); iter != elements.end(); ++iter)
  947. {
  948. GUIElement* element = *iter;
  949. if(!element->_isDisabled() && element->_isInBounds(localPos))
  950. {
  951. mNewElementsUnderPointer.push_back(ElementInfo(element, widget));
  952. }
  953. }
  954. }
  955. widgetIdx++;
  956. }
  957. }
  958. std::sort(mNewElementsUnderPointer.begin(), mNewElementsUnderPointer.end(),
  959. [](const ElementInfo& a, const ElementInfo& b)
  960. {
  961. return a.element->_getDepth() < b.element->_getDepth();
  962. });
  963. // Send MouseOut and MouseOver events
  964. bool eventProcessed = false;
  965. for(auto& elementInfo : mElementsUnderPointer)
  966. {
  967. GUIElement* element = elementInfo.element;
  968. GUIWidget* widget = elementInfo.widget;
  969. auto iterFind = std::find_if(mNewElementsUnderPointer.begin(), mNewElementsUnderPointer.end(),
  970. [=] (const ElementInfo& x) { return x.element == element; });
  971. if(iterFind == mNewElementsUnderPointer.end())
  972. {
  973. auto iterFind2 = std::find_if(mActiveElements.begin(), mActiveElements.end(),
  974. [=](const ElementInfo& x) { return x.element == element; });
  975. // Send MouseOut event
  976. if(mActiveElements.size() == 0 || iterFind2 != mActiveElements.end())
  977. {
  978. Vector2I localPos = getWidgetRelativePos(*widget, pointerScreenPos);
  979. mMouseEvent.setMouseOutData(localPos);
  980. if(sendMouseEvent(widget, element, mMouseEvent))
  981. eventProcessed = true;
  982. }
  983. }
  984. }
  985. for(auto& elementInfo : mNewElementsUnderPointer)
  986. {
  987. GUIElement* element = elementInfo.element;
  988. GUIWidget* widget = elementInfo.widget;
  989. auto iterFind = std::find_if(begin(mElementsUnderPointer), end(mElementsUnderPointer),
  990. [=] (const ElementInfo& x) { return x.element == element; });
  991. if(iterFind == mElementsUnderPointer.end())
  992. {
  993. auto iterFind2 = std::find_if(mActiveElements.begin(), mActiveElements.end(),
  994. [&](const ElementInfo& x) { return x.element == element; });
  995. // Send MouseOver event
  996. if(mActiveElements.size() == 0 || iterFind2 != mActiveElements.end())
  997. {
  998. Vector2I localPos;
  999. if(widget != nullptr)
  1000. localPos = getWidgetRelativePos(*widget, pointerScreenPos);
  1001. mMouseEvent = GUIMouseEvent(buttonStates, shift, control, alt);
  1002. mMouseEvent.setMouseOverData(localPos);
  1003. if(sendMouseEvent(widget, element, mMouseEvent))
  1004. eventProcessed = true;
  1005. }
  1006. }
  1007. }
  1008. mElementsUnderPointer.swap(mNewElementsUnderPointer);
  1009. return eventProcessed;
  1010. }
  1011. void GUIManager::onTextInput(const TextInputEvent& event)
  1012. {
  1013. mTextInputEvent = GUITextInputEvent();
  1014. mTextInputEvent.setData(event.textChar);
  1015. for(auto& elementInFocus : mElementsInFocus)
  1016. {
  1017. if(sendTextInputEvent(elementInFocus.widget, elementInFocus.element, mTextInputEvent))
  1018. event.markAsUsed();
  1019. }
  1020. }
  1021. void GUIManager::onWindowFocusGained(RenderWindow& win)
  1022. {
  1023. for(auto& widgetInfo : mWidgets)
  1024. {
  1025. GUIWidget* widget = widgetInfo.widget;
  1026. if(getWidgetWindow(*widget) == &win)
  1027. widget->ownerWindowFocusChanged();
  1028. }
  1029. }
  1030. void GUIManager::onWindowFocusLost(RenderWindow& win)
  1031. {
  1032. for(auto& widgetInfo : mWidgets)
  1033. {
  1034. GUIWidget* widget = widgetInfo.widget;
  1035. if(getWidgetWindow(*widget) == &win)
  1036. widget->ownerWindowFocusChanged();
  1037. }
  1038. mNewElementsInFocus.clear();
  1039. for(auto& focusedElement : mElementsInFocus)
  1040. {
  1041. if(getWidgetWindow(*focusedElement.widget) == &win)
  1042. {
  1043. mCommandEvent = GUICommandEvent();
  1044. mCommandEvent.setType(GUICommandEventType::FocusLost);
  1045. sendCommandEvent(focusedElement.widget, focusedElement.element, mCommandEvent);
  1046. }
  1047. else
  1048. mNewElementsInFocus.push_back(focusedElement);
  1049. }
  1050. mElementsInFocus.swap(mNewElementsInFocus);
  1051. }
  1052. // We stop getting mouse move events once it leaves the window, so make sure
  1053. // nothing stays in hover state
  1054. void GUIManager::onMouseLeftWindow(RenderWindow* win)
  1055. {
  1056. bool buttonStates[3];
  1057. buttonStates[0] = false;
  1058. buttonStates[1] = false;
  1059. buttonStates[2] = false;
  1060. mNewElementsUnderPointer.clear();
  1061. for(auto& elementInfo : mElementsUnderPointer)
  1062. {
  1063. GUIElement* element = elementInfo.element;
  1064. GUIWidget* widget = elementInfo.widget;
  1065. if(widget->getTarget()->getTarget().get() != win)
  1066. {
  1067. mNewElementsUnderPointer.push_back(elementInfo);
  1068. continue;
  1069. }
  1070. auto iterFind = std::find_if(mActiveElements.begin(), mActiveElements.end(),
  1071. [&](const ElementInfo& x) { return x.element == element; });
  1072. // Send MouseOut event
  1073. if(mActiveElements.size() == 0 || iterFind != mActiveElements.end())
  1074. {
  1075. Vector2I curLocalPos = getWidgetRelativePos(*widget, Vector2I());
  1076. mMouseEvent.setMouseOutData(curLocalPos);
  1077. sendMouseEvent(widget, element, mMouseEvent);
  1078. }
  1079. }
  1080. mElementsUnderPointer.swap(mNewElementsUnderPointer);
  1081. if(mDragState != DragState::Dragging)
  1082. {
  1083. if(mActiveCursor != CursorType::Arrow)
  1084. {
  1085. Cursor::instance().setCursor(CursorType::Arrow);
  1086. mActiveCursor = CursorType::Arrow;
  1087. }
  1088. }
  1089. }
  1090. void GUIManager::queueForDestroy(GUIElement* element)
  1091. {
  1092. mScheduledForDestruction.push(element);
  1093. }
  1094. void GUIManager::setFocus(GUIElement* element, bool focus)
  1095. {
  1096. ElementFocusInfo efi;
  1097. efi.element = element;
  1098. efi.focus = focus;
  1099. mForcedFocusElements.push_back(efi);
  1100. }
  1101. void GUIManager::processDestroyQueue()
  1102. {
  1103. // Need two loops and a temporary since element destructors may themselves
  1104. // queue other elements for destruction
  1105. while(!mScheduledForDestruction.empty())
  1106. {
  1107. Stack<GUIElement*> toDestroy = mScheduledForDestruction;
  1108. mScheduledForDestruction = Stack<GUIElement*>();
  1109. while(!toDestroy.empty())
  1110. {
  1111. bs_delete<PoolAlloc>(toDestroy.top());
  1112. toDestroy.pop();
  1113. }
  1114. }
  1115. }
  1116. void GUIManager::setInputBridge(const RenderTexture* renderTex, const GUIElement* element)
  1117. {
  1118. if(element == nullptr)
  1119. mInputBridge.erase(renderTex);
  1120. else
  1121. mInputBridge[renderTex] = element;
  1122. }
  1123. GUIMouseButton GUIManager::buttonToGUIButton(PointerEventButton pointerButton) const
  1124. {
  1125. if(pointerButton == PointerEventButton::Left)
  1126. return GUIMouseButton::Left;
  1127. else if(pointerButton == PointerEventButton::Middle)
  1128. return GUIMouseButton::Middle;
  1129. else if(pointerButton == PointerEventButton::Right)
  1130. return GUIMouseButton::Right;
  1131. BS_EXCEPT(InvalidParametersException, "Provided button is not a GUI supported mouse button.");
  1132. }
  1133. Vector2I GUIManager::getWidgetRelativePos(const GUIWidget& widget, const Vector2I& screenPos) const
  1134. {
  1135. const RenderWindow* window = getWidgetWindow(widget);
  1136. if(window == nullptr)
  1137. return Vector2I();
  1138. Vector2I windowPos = window->screenToWindowPos(screenPos);
  1139. windowPos = windowToBridgedCoords(widget, windowPos);
  1140. const Matrix4& worldTfrm = widget.SO()->getWorldTfrm();
  1141. Vector4 vecLocalPos = worldTfrm.inverse().multiply3x4(Vector4((float)windowPos.x, (float)windowPos.y, 0.0f, 1.0f));
  1142. Vector2I curLocalPos(Math::roundToInt(vecLocalPos.x), Math::roundToInt(vecLocalPos.y));
  1143. return curLocalPos;
  1144. }
  1145. Vector2I GUIManager::windowToBridgedCoords(const GUIWidget& widget, const Vector2I& windowPos) const
  1146. {
  1147. // This cast might not be valid (the render target could be a window), but we only really need to cast
  1148. // so that mInputBridge map allows us to search through it - we don't access anything unless the target is bridged
  1149. // (in which case we know it is a RenderTexture)
  1150. const RenderTexture* renderTexture = static_cast<const RenderTexture*>(widget.getTarget()->getTarget().get());
  1151. auto iterFind = mInputBridge.find(renderTexture);
  1152. if(iterFind != mInputBridge.end()) // Widget input is bridged, which means we need to transform the coordinates
  1153. {
  1154. const GUIElement* bridgeElement = iterFind->second;
  1155. const Matrix4& worldTfrm = bridgeElement->_getParentWidget()->SO()->getWorldTfrm();
  1156. Vector4 vecLocalPos = worldTfrm.inverse().multiply3x4(Vector4((float)windowPos.x, (float)windowPos.y, 0.0f, 1.0f));
  1157. RectI bridgeBounds = bridgeElement->getBounds();
  1158. // Find coordinates relative to the bridge element
  1159. float x = vecLocalPos.x - (float)bridgeBounds.x;
  1160. float y = vecLocalPos.y - (float)bridgeBounds.y;
  1161. float scaleX = renderTexture->getWidth() / (float)bridgeBounds.width;
  1162. float scaleY = renderTexture->getHeight() / (float)bridgeBounds.height;
  1163. return Vector2I(Math::roundToInt(x * scaleX), Math::roundToInt(y * scaleY));
  1164. }
  1165. return windowPos;
  1166. }
  1167. const RenderWindow* GUIManager::getWidgetWindow(const GUIWidget& widget) const
  1168. {
  1169. // This cast might not be valid (the render target could be a window), but we only really need to cast
  1170. // so that mInputBridge map allows us to search through it - we don't access anything unless the target is bridged
  1171. // (in which case we know it is a RenderTexture)
  1172. const RenderTexture* renderTexture = static_cast<const RenderTexture*>(widget.getTarget()->getTarget().get());
  1173. auto iterFind = mInputBridge.find(renderTexture);
  1174. if(iterFind != mInputBridge.end())
  1175. {
  1176. GUIWidget* parentWidget = iterFind->second->_getParentWidget();
  1177. if(parentWidget != &widget)
  1178. {
  1179. return getWidgetWindow(*parentWidget);
  1180. }
  1181. }
  1182. RenderTargetPtr renderTarget = widget.getTarget()->getTarget();
  1183. Vector<RenderWindow*> renderWindows = RenderWindowManager::instance().getRenderWindows();
  1184. auto iterFindWin = std::find(renderWindows.begin(), renderWindows.end(), renderTarget.get());
  1185. if(iterFindWin != renderWindows.end())
  1186. return static_cast<RenderWindow*>(renderTarget.get());
  1187. return nullptr;
  1188. }
  1189. bool GUIManager::sendMouseEvent(GUIWidget* widget, GUIElement* element, const GUIMouseEvent& event)
  1190. {
  1191. return widget->_mouseEvent(element, event);
  1192. }
  1193. bool GUIManager::sendTextInputEvent(GUIWidget* widget, GUIElement* element, const GUITextInputEvent& event)
  1194. {
  1195. return widget->_textInputEvent(element, event);
  1196. }
  1197. bool GUIManager::sendCommandEvent(GUIWidget* widget, GUIElement* element, const GUICommandEvent& event)
  1198. {
  1199. return widget->_commandEvent(element, event);
  1200. }
  1201. bool GUIManager::sendVirtualButtonEvent(GUIWidget* widget, GUIElement* element, const GUIVirtualButtonEvent& event)
  1202. {
  1203. return widget->_virtualButtonEvent(element, event);
  1204. }
  1205. GUIManager& gGUIManager()
  1206. {
  1207. return GUIManager::instance();
  1208. }
  1209. }