#include "BsEditorWidgetContainer.h" #include "BsGUITabbedTitleBar.h" #include "BsEditorWidget.h" #include "BsDragAndDropManager.h" #include "BsEditorWindow.h" #include "BsGUIArea.h" #include "CmMath.h" #include "CmInput.h" #include "BsGUIWidget.h" #include "BsGUILayout.h" using namespace CamelotFramework; using namespace BansheeEngine; namespace BansheeEditor { const CM::UINT32 EditorWidgetContainer::TitleBarHeight = 13; EditorWidgetContainer::EditorWidgetContainer(BS::GUIWidget* parent, RenderWindow* renderWindow) :mParent(parent), mX(0), mY(0), mWidth(0), mHeight(0), mTitleBar(nullptr), mActiveWidget(-1), mTitleBarArea(nullptr) { mTitleBarArea = GUIArea::create(*parent, 0, 0, 0, 0, 9900); mTitleBar = GUITabbedTitleBar::create(*parent, renderWindow); mTitleBar->onTabActivated.connect(boost::bind(&EditorWidgetContainer::tabActivated, this, _1)); mTitleBar->onTabClosed.connect(boost::bind(&EditorWidgetContainer::tabClosed, this, _1)); mTitleBar->onTabDraggedOff.connect(boost::bind(&EditorWidgetContainer::tabDraggedOff, this, _1)); mTitleBar->onTabDraggedOn.connect(boost::bind(&EditorWidgetContainer::tabDraggedOn, this, _1)); mTitleBarArea->getLayout().addElement(mTitleBar); } EditorWidgetContainer::~EditorWidgetContainer() { GUIArea::destroy(mTitleBarArea); GUIElement::destroy(mTitleBar); for(auto& widget : mWidgets) { EditorWidgetBase::destroy(widget.second); } } void EditorWidgetContainer::add(EditorWidgetBase& widget) { insert((UINT32)mWidgets.size(), widget); } void EditorWidgetContainer::remove(EditorWidgetBase& widget) { INT32 tabIdx = 0; for(auto& curWidget : mWidgets) { if(curWidget.second == &widget) { tabIdx = curWidget.first; break; } } if(tabIdx == -1) return; mWidgets.erase((UINT32)tabIdx); mTitleBar->removeTab((UINT32)tabIdx); widget._changeParent(nullptr); if(tabIdx == mActiveWidget) { if(mTitleBar->getNumTabs() > 0) { mActiveWidget = -1; setActiveWidget(mTitleBar->getTabIdx(0)); } } } void EditorWidgetContainer::insert(CM::UINT32 idx, EditorWidgetBase& widget) { for(auto& curWidget : mWidgets) { if(curWidget.second == &widget) return; } idx = Math::clamp(idx, 0U, (UINT32)mWidgets.size()); UINT32 tabIdx = mTitleBar->insertTab(idx, widget.getName()); mWidgets[tabIdx] = &widget; widget._changeParent(this); if(mActiveWidget == -1) setActiveWidget(mTitleBar->getTabIdx(mTitleBar->getNumTabs() - 1)); else widget._disable(); } void EditorWidgetContainer::setSize(UINT32 width, UINT32 height) { // TODO - Title bar is always TitleBarHeight size, so what happens when the container area is smaller than that? mTitleBarArea->setSize(width, TitleBarHeight); if(mActiveWidget >= 0) { EditorWidgetBase* activeWidgetPtr = mWidgets[mActiveWidget]; UINT32 contentHeight = (UINT32)std::max(0, (INT32)height - (INT32)TitleBarHeight); activeWidgetPtr->_setSize(width, contentHeight); } mWidth = width; mHeight = height; } void EditorWidgetContainer::setPosition(INT32 x, INT32 y) { mTitleBarArea->setPosition(x, y); if(mActiveWidget >= 0) { EditorWidgetBase* activeWidgetPtr = mWidgets[mActiveWidget]; activeWidgetPtr->_setPosition(x, y + TitleBarHeight); } mX = x; mY = y; } void EditorWidgetContainer::setActiveWidget(UINT32 idx) { if(mActiveWidget == idx) return; mActiveWidget = idx; for(auto& curWidget : mWidgets) { if(curWidget.first != (UINT32)mActiveWidget) curWidget.second->_disable(); else curWidget.second->_enable(); } mTitleBar->setActive(idx); setPosition(mX, mY); setSize(mWidth, mHeight); } void EditorWidgetContainer::tabActivated(UINT32 uniqueIdx) { setActiveWidget(uniqueIdx); } void EditorWidgetContainer::tabClosed(UINT32 uniqueIdx) { EditorWidgetBase* widget = mWidgets[uniqueIdx]; remove(*widget); EditorWidgetBase::destroy(widget); if(!onWidgetClosed.empty()) onWidgetClosed(); } void EditorWidgetContainer::tabDraggedOff(CM::UINT32 uniqueIdx) { EditorWidgetBase* widget = mWidgets[uniqueIdx]; remove(*widget); // TODO - Hook up drag and drop texture DragAndDropManager::instance().startDrag(HTexture(), (UINT32)DragAndDropType::EditorWidget, (void*)widget, boost::bind(&EditorWidgetContainer::tabDroppedCallback, _1)); if(!onWidgetClosed.empty()) onWidgetClosed(); } void EditorWidgetContainer::tabDraggedOn(CM::UINT32 seqIdx) { #if CM_DEBUG_MODE if(!DragAndDropManager::instance().isDragInProgress()) CM_EXCEPT(InternalErrorException, "Tab drag and drop reported but no drag in progress."); if(DragAndDropManager::instance().getDragTypeId() != (UINT32)DragAndDropType::EditorWidget) CM_EXCEPT(InternalErrorException, "Tab drag and drop reported but drag type is invalid."); #endif EditorWidgetBase* draggedWidget = static_cast(DragAndDropManager::instance().getDragData()); insert(seqIdx, *draggedWidget); setActiveWidget(mTitleBar->getTabIdx(seqIdx)); } void EditorWidgetContainer::tabDroppedCallback(bool wasDragProcessed) { if(DragAndDropManager::instance().getDragTypeId() != (UINT32)DragAndDropType::EditorWidget) return; if(!wasDragProcessed) { EditorWidgetBase* draggedWidget = static_cast(DragAndDropManager::instance().getDragData()); EditorWindow* newWindow = EditorWindow::create(); newWindow->widgets().add(*draggedWidget); Vector2I mousePos = Input::instance().getCursorPosition(); newWindow->setPosition(mousePos.x, mousePos.y); } } RectI EditorWidgetContainer::getContentBounds() const { return RectI(mX, mY + TitleBarHeight, mWidth, (UINT32)std::max(0, (INT32)mHeight - (INT32)TitleBarHeight)); } void EditorWidgetContainer::_notifyWidgetDestroyed(EditorWidgetBase* widget) { for(auto& curWidget : mWidgets) { if(curWidget.second == widget) { remove(*widget); if(!onWidgetClosed.empty()) onWidgetClosed(); return; } } } }