BsEditorWidgetContainer.cpp 5.9 KB

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