BsEditorWidgetContainer.cpp 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. #include "BsEditorWidgetContainer.h"
  2. #include "BsGUITabbedTitleBar.h"
  3. #include "BsEditorWidget.h"
  4. #include "BsDragAndDropManager.h"
  5. #include "BsEditorWindow.h"
  6. #include "CmMath.h"
  7. #include "CmInput.h"
  8. using namespace CamelotFramework;
  9. using namespace BansheeEngine;
  10. namespace BansheeEditor
  11. {
  12. const CM::UINT32 EditorWidgetContainer::TitleBarHeight = 13;
  13. EditorWidgetContainer::EditorWidgetContainer(BS::GUIWidget* parent)
  14. :mParent(parent), mX(0), mY(0), mWidth(0), mHeight(0), mTitleBar(nullptr), mActiveWidget(-1)
  15. {
  16. mTitleBar = cm_new<GUITabbedTitleBar>(parent);
  17. mTitleBar->onTabActivated.connect(boost::bind(&EditorWidgetContainer::tabActivated, this, _1));
  18. mTitleBar->onTabClosed.connect(boost::bind(&EditorWidgetContainer::tabClosed, this, _1));
  19. mTitleBar->onTabDraggedOff.connect(boost::bind(&EditorWidgetContainer::tabDraggedOff, this, _1));
  20. mTitleBar->onTabDraggedOn.connect(boost::bind(&EditorWidgetContainer::tabDraggedOn, this, _1));
  21. }
  22. EditorWidgetContainer::~EditorWidgetContainer()
  23. {
  24. cm_delete(mTitleBar);
  25. for(auto& widget : mWidgets)
  26. {
  27. EditorWidget::destroy(widget);
  28. }
  29. }
  30. void EditorWidgetContainer::add(EditorWidget& widget)
  31. {
  32. insert((UINT32)mWidgets.size(), widget);
  33. }
  34. void EditorWidgetContainer::remove(EditorWidget& widget)
  35. {
  36. INT32 widgetIdx = -1;
  37. UINT32 curIdx = 0;
  38. for(auto& curWidget : mWidgets)
  39. {
  40. if(curWidget == &widget)
  41. {
  42. widgetIdx = curIdx;
  43. break;
  44. }
  45. curIdx++;
  46. }
  47. if(widgetIdx == -1)
  48. return;
  49. mWidgets.erase(mWidgets.begin() + widgetIdx);
  50. mTitleBar->removeTab((UINT32)widgetIdx);
  51. widget._changeParent(nullptr);
  52. if(widgetIdx == mActiveWidget)
  53. {
  54. if(mWidgets.size() > 0)
  55. {
  56. setActiveWidget(0);
  57. }
  58. }
  59. }
  60. void EditorWidgetContainer::insert(CM::UINT32 idx, EditorWidget& widget)
  61. {
  62. for(auto& curWidget : mWidgets)
  63. {
  64. if(curWidget == &widget)
  65. return;
  66. }
  67. idx = Math::Clamp(idx, 0U, (UINT32)mWidgets.size());
  68. mTitleBar->insertTab(idx, widget.getName());
  69. mWidgets.insert(mWidgets.begin() + idx, &widget);
  70. widget._changeParent(this);
  71. if(mActiveWidget == -1)
  72. setActiveWidget((UINT32)mWidgets.size() - 1);
  73. else
  74. widget._disable();
  75. }
  76. void EditorWidgetContainer::setSize(UINT32 width, UINT32 height)
  77. {
  78. // TODO - Title bar is always TitleBarHeight size, so what happens when the container area is smaller than that?
  79. mTitleBar->setSize(width, TitleBarHeight);
  80. if(mActiveWidget >= 0)
  81. {
  82. EditorWidget* activeWidgetPtr = mWidgets[mActiveWidget];
  83. UINT32 contentHeight = (UINT32)std::max(0, (INT32)height - (INT32)TitleBarHeight);
  84. activeWidgetPtr->_setSize(width, contentHeight);
  85. }
  86. mWidth = width;
  87. mHeight = height;
  88. }
  89. void EditorWidgetContainer::setPosition(INT32 x, INT32 y)
  90. {
  91. mTitleBar->setPosition(x, y);
  92. if(mActiveWidget >= 0)
  93. {
  94. EditorWidget* activeWidgetPtr = mWidgets[mActiveWidget];
  95. activeWidgetPtr->_setPosition(x, y + TitleBarHeight);
  96. }
  97. mX = x;
  98. mY = y;
  99. }
  100. void EditorWidgetContainer::setActiveWidget(UINT32 idx)
  101. {
  102. if(mActiveWidget == idx)
  103. return;
  104. mActiveWidget = idx;
  105. UINT32 curIdx = 0;
  106. for(auto& curWidget : mWidgets)
  107. {
  108. if(curIdx != (UINT32)mActiveWidget)
  109. curWidget->_disable();
  110. else
  111. curWidget->_enable();
  112. curIdx++;
  113. }
  114. setPosition(mX, mY);
  115. setSize(mWidth, mHeight);
  116. }
  117. void EditorWidgetContainer::tabActivated(UINT32 idx)
  118. {
  119. setActiveWidget(idx);
  120. }
  121. void EditorWidgetContainer::tabClosed(UINT32 idx)
  122. {
  123. EditorWidget* widget = mWidgets[idx];
  124. remove(*widget);
  125. EditorWidget::destroy(widget);
  126. if(!onWidgetClosed.empty())
  127. onWidgetClosed();
  128. }
  129. void EditorWidgetContainer::tabDraggedOff(CM::UINT32 idx)
  130. {
  131. EditorWidget* widget = mWidgets[idx];
  132. remove(*widget);
  133. // TODO - Hook up drag and drop texture
  134. DragAndDropManager::instance().startDrag(HTexture(), (UINT32)DragAndDropType::EditorWidget, (void*)widget, boost::bind(&EditorWidgetContainer::tabDroppedCallback, this, _1));
  135. if(!onWidgetClosed.empty())
  136. onWidgetClosed();
  137. }
  138. void EditorWidgetContainer::tabDraggedOn(CM::UINT32 idx)
  139. {
  140. #if CM_DEBUG_MODE
  141. if(!DragAndDropManager::instance().isDragInProgress())
  142. CM_EXCEPT(InternalErrorException, "Tab drag and drop reported but no drag in progress.");
  143. if(DragAndDropManager::instance().getDragTypeId() != (UINT32)DragAndDropType::EditorWidget)
  144. CM_EXCEPT(InternalErrorException, "Tab drag and drop reported but drag type is invalid.");
  145. #endif
  146. EditorWidget* draggedWidget = static_cast<EditorWidget*>(DragAndDropManager::instance().getDragData());
  147. insert(idx, *draggedWidget);
  148. }
  149. void EditorWidgetContainer::tabDroppedCallback(bool wasDragProcessed)
  150. {
  151. if(!wasDragProcessed)
  152. {
  153. EditorWindow* newWindow = EditorWindow::create();
  154. EditorWidget* draggedWidget = static_cast<EditorWidget*>(DragAndDropManager::instance().getDragData());
  155. newWindow->widgets().add(*draggedWidget);
  156. Int2 mousePos = Input::instance().getMousePosition();
  157. newWindow->setPosition(mousePos.x, mousePos.y);
  158. }
  159. }
  160. void EditorWidgetContainer::_notifyWidgetDestroyed(EditorWidget* widget)
  161. {
  162. for(auto& curWidget : mWidgets)
  163. {
  164. if(curWidget == widget)
  165. {
  166. remove(*widget);
  167. if(!onWidgetClosed.empty())
  168. onWidgetClosed();
  169. return;
  170. }
  171. }
  172. }
  173. }