2015-02-12 23:16:25 +08:00
|
|
|
// Aseprite
|
|
|
|
// Copyright (C) 2001-2015 David Capello
|
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License version 2 as
|
|
|
|
// published by the Free Software Foundation.
|
2013-01-21 05:40:37 +08:00
|
|
|
|
2013-08-06 08:20:19 +08:00
|
|
|
#ifdef HAVE_CONFIG_H
|
2013-01-21 05:40:37 +08:00
|
|
|
#include "config.h"
|
2013-08-06 08:20:19 +08:00
|
|
|
#endif
|
2013-01-21 05:40:37 +08:00
|
|
|
|
2013-08-06 08:20:19 +08:00
|
|
|
#include "app/ui/workspace.h"
|
2013-01-21 05:40:37 +08:00
|
|
|
|
2015-03-05 04:23:40 +08:00
|
|
|
#include "app/ui/skin/skin_theme.h"
|
2015-04-03 02:43:50 +08:00
|
|
|
#include "app/ui/workspace_tabs.h"
|
2013-08-06 08:20:19 +08:00
|
|
|
#include "app/ui/workspace_view.h"
|
2015-03-05 04:23:40 +08:00
|
|
|
#include "base/remove_from_container.h"
|
2015-04-01 04:31:45 +08:00
|
|
|
#include "ui/paint_event.h"
|
|
|
|
#include "ui/resize_event.h"
|
2013-01-21 05:40:37 +08:00
|
|
|
|
2013-08-06 08:20:19 +08:00
|
|
|
namespace app {
|
2013-01-21 05:40:37 +08:00
|
|
|
|
2013-08-06 08:20:19 +08:00
|
|
|
using namespace app::skin;
|
|
|
|
using namespace ui;
|
2015-02-12 23:16:25 +08:00
|
|
|
|
2015-04-01 04:31:45 +08:00
|
|
|
// static
|
|
|
|
WidgetType Workspace::Type()
|
|
|
|
{
|
|
|
|
static WidgetType type = kGenericWidget;
|
|
|
|
if (type == kGenericWidget)
|
|
|
|
type = register_widget_type();
|
|
|
|
return type;
|
|
|
|
}
|
|
|
|
|
2013-01-21 05:40:37 +08:00
|
|
|
Workspace::Workspace()
|
2015-04-01 04:31:45 +08:00
|
|
|
: Widget(Workspace::Type())
|
|
|
|
, m_mainPanel(WorkspacePanel::MAIN_PANEL)
|
|
|
|
, m_tabs(nullptr)
|
|
|
|
, m_activePanel(&m_mainPanel)
|
|
|
|
, m_dropPreviewPanel(nullptr)
|
2015-04-03 02:43:50 +08:00
|
|
|
, m_dropPreviewTabs(nullptr)
|
2013-01-21 05:40:37 +08:00
|
|
|
{
|
|
|
|
SkinTheme* theme = static_cast<SkinTheme*>(getTheme());
|
2015-02-16 02:29:16 +08:00
|
|
|
setBgColor(theme->colors.workspace());
|
2015-04-01 04:31:45 +08:00
|
|
|
|
|
|
|
addChild(&m_mainPanel);
|
2013-01-21 05:40:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Workspace::~Workspace()
|
|
|
|
{
|
2013-03-28 08:19:35 +08:00
|
|
|
// No views at this point.
|
|
|
|
ASSERT(m_views.empty());
|
2013-01-21 05:40:37 +08:00
|
|
|
}
|
|
|
|
|
2015-04-03 02:43:50 +08:00
|
|
|
void Workspace::setTabsBar(WorkspaceTabs* tabs)
|
2015-03-28 04:26:04 +08:00
|
|
|
{
|
2015-04-01 04:31:45 +08:00
|
|
|
m_tabs = tabs;
|
|
|
|
m_mainPanel.setTabsBar(tabs);
|
2015-03-28 04:26:04 +08:00
|
|
|
}
|
|
|
|
|
2015-03-19 04:34:22 +08:00
|
|
|
void Workspace::addView(WorkspaceView* view, int pos)
|
2013-01-21 05:40:37 +08:00
|
|
|
{
|
2015-04-05 03:45:58 +08:00
|
|
|
addViewToPanel(&m_mainPanel, view, false, pos);
|
2013-01-21 05:40:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void Workspace::removeView(WorkspaceView* view)
|
|
|
|
{
|
2015-03-05 04:23:40 +08:00
|
|
|
base::remove_from_container(m_views, view);
|
2013-01-21 05:40:37 +08:00
|
|
|
|
2015-04-01 04:31:45 +08:00
|
|
|
WorkspacePanel* panel = getViewPanel(view);
|
|
|
|
ASSERT(panel);
|
|
|
|
if (panel)
|
|
|
|
panel->removeView(view);
|
2013-03-28 08:19:35 +08:00
|
|
|
}
|
|
|
|
|
2015-02-23 08:18:53 +08:00
|
|
|
bool Workspace::closeView(WorkspaceView* view)
|
|
|
|
{
|
|
|
|
return view->onCloseView(this);
|
|
|
|
}
|
|
|
|
|
2014-07-29 11:53:24 +08:00
|
|
|
WorkspaceView* Workspace::activeView()
|
2013-03-28 08:19:35 +08:00
|
|
|
{
|
2015-04-01 04:31:45 +08:00
|
|
|
return (m_activePanel ? m_activePanel->activeView(): nullptr);
|
2013-01-21 05:40:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void Workspace::setActiveView(WorkspaceView* view)
|
|
|
|
{
|
2015-04-01 04:31:45 +08:00
|
|
|
m_activePanel = getViewPanel(view);
|
|
|
|
ASSERT(m_activePanel);
|
|
|
|
if (!m_activePanel)
|
|
|
|
return;
|
2013-01-21 05:40:37 +08:00
|
|
|
|
2015-04-01 04:31:45 +08:00
|
|
|
m_activePanel->setActiveView(view);
|
2013-03-28 08:19:35 +08:00
|
|
|
|
2015-04-01 04:31:45 +08:00
|
|
|
ActiveViewChanged(); // Fire ActiveViewChanged event
|
|
|
|
}
|
2013-03-28 08:19:35 +08:00
|
|
|
|
2015-04-01 04:31:45 +08:00
|
|
|
void Workspace::setMainPanelAsActive()
|
|
|
|
{
|
|
|
|
m_activePanel = &m_mainPanel;
|
2015-04-01 22:35:52 +08:00
|
|
|
|
|
|
|
removeDropViewPreview();
|
|
|
|
m_dropPreviewPanel = nullptr;
|
2015-04-03 03:02:24 +08:00
|
|
|
m_dropPreviewTabs = nullptr;
|
2015-04-01 22:35:52 +08:00
|
|
|
|
2013-03-28 08:19:35 +08:00
|
|
|
ActiveViewChanged(); // Fire ActiveViewChanged event
|
2013-01-21 05:40:37 +08:00
|
|
|
}
|
|
|
|
|
2015-04-17 20:49:49 +08:00
|
|
|
bool Workspace::canSelectOtherTab() const
|
|
|
|
{
|
|
|
|
return m_activePanel->tabs()->canSelectOtherTab();
|
|
|
|
}
|
|
|
|
|
2015-04-05 01:40:07 +08:00
|
|
|
void Workspace::selectNextTab()
|
|
|
|
{
|
|
|
|
m_activePanel->tabs()->selectNextTab();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Workspace::selectPreviousTab()
|
|
|
|
{
|
|
|
|
m_activePanel->tabs()->selectPreviousTab();
|
|
|
|
}
|
|
|
|
|
2015-04-06 23:22:20 +08:00
|
|
|
void Workspace::duplicateActiveView()
|
|
|
|
{
|
|
|
|
WorkspaceView* view = activeView();
|
|
|
|
if (!view)
|
|
|
|
return;
|
|
|
|
|
|
|
|
WorkspaceView* clone = view->cloneWorkspaceView();
|
|
|
|
if (!clone)
|
|
|
|
return;
|
|
|
|
|
|
|
|
WorkspacePanel* panel = getViewPanel(view);
|
|
|
|
addViewToPanel(panel, clone, false, -1);
|
|
|
|
clone->onClonedFrom(view);
|
|
|
|
setActiveView(clone);
|
|
|
|
}
|
|
|
|
|
2015-03-05 04:23:40 +08:00
|
|
|
void Workspace::onPaint(PaintEvent& ev)
|
2014-01-26 22:22:23 +08:00
|
|
|
{
|
2015-03-05 04:23:40 +08:00
|
|
|
ev.getGraphics()->fillRect(getBgColor(), getClientBounds());
|
2014-01-26 22:22:23 +08:00
|
|
|
}
|
|
|
|
|
2015-03-28 05:14:00 +08:00
|
|
|
void Workspace::onResize(ui::ResizeEvent& ev)
|
|
|
|
{
|
|
|
|
setBoundsQuietly(ev.getBounds());
|
|
|
|
|
2015-03-28 05:45:36 +08:00
|
|
|
gfx::Rect rc = getChildrenBounds();
|
2015-03-28 05:14:00 +08:00
|
|
|
for (Widget* child : getChildren())
|
2015-03-28 05:45:36 +08:00
|
|
|
child->setBounds(rc);
|
2015-03-28 05:14:00 +08:00
|
|
|
}
|
|
|
|
|
2015-04-05 03:25:57 +08:00
|
|
|
DropViewPreviewResult Workspace::setDropViewPreview(const gfx::Point& pos,
|
2015-04-03 02:43:50 +08:00
|
|
|
WorkspaceView* view, WorkspaceTabs* tabs)
|
2015-03-28 05:14:00 +08:00
|
|
|
{
|
2015-04-03 02:43:50 +08:00
|
|
|
TabView* tabView = dynamic_cast<TabView*>(view);
|
|
|
|
WorkspaceTabs* newTabs = nullptr;
|
2015-04-01 04:31:45 +08:00
|
|
|
WorkspacePanel* panel = getPanelAt(pos);
|
2015-04-03 02:43:50 +08:00
|
|
|
if (!newTabs) {
|
|
|
|
newTabs = getTabsAt(pos);
|
|
|
|
// Drop preview is only to drop tabs from a different WorkspaceTabs.
|
|
|
|
if (newTabs == tabs)
|
|
|
|
newTabs = nullptr;
|
|
|
|
}
|
2015-03-28 05:14:00 +08:00
|
|
|
|
2015-04-01 04:31:45 +08:00
|
|
|
if (m_dropPreviewPanel && m_dropPreviewPanel != panel)
|
|
|
|
m_dropPreviewPanel->removeDropViewPreview();
|
2015-04-03 02:43:50 +08:00
|
|
|
if (m_dropPreviewTabs && m_dropPreviewTabs != newTabs)
|
|
|
|
m_dropPreviewTabs->removeDropViewPreview();
|
2015-03-28 06:42:18 +08:00
|
|
|
|
2015-04-01 04:31:45 +08:00
|
|
|
m_dropPreviewPanel = panel;
|
2015-04-03 02:43:50 +08:00
|
|
|
m_dropPreviewTabs = newTabs;
|
2015-03-28 06:42:18 +08:00
|
|
|
|
2015-04-01 04:31:45 +08:00
|
|
|
if (m_dropPreviewPanel)
|
2015-04-03 02:43:50 +08:00
|
|
|
m_dropPreviewPanel->setDropViewPreview(pos, view);
|
|
|
|
if (m_dropPreviewTabs)
|
|
|
|
m_dropPreviewTabs->setDropViewPreview(pos, tabView);
|
2015-04-05 03:25:57 +08:00
|
|
|
|
|
|
|
if (panel)
|
|
|
|
return DropViewPreviewResult::DROP_IN_PANEL;
|
|
|
|
else if (newTabs)
|
|
|
|
return DropViewPreviewResult::DROP_IN_TABS;
|
|
|
|
else
|
|
|
|
return DropViewPreviewResult::FLOATING;
|
2015-03-28 05:14:00 +08:00
|
|
|
}
|
|
|
|
|
2015-04-01 04:31:45 +08:00
|
|
|
void Workspace::removeDropViewPreview()
|
2015-03-28 06:42:18 +08:00
|
|
|
{
|
2015-04-05 01:53:27 +08:00
|
|
|
if (m_dropPreviewPanel) {
|
2015-04-01 04:31:45 +08:00
|
|
|
m_dropPreviewPanel->removeDropViewPreview();
|
2015-04-05 01:53:27 +08:00
|
|
|
m_dropPreviewPanel = nullptr;
|
|
|
|
}
|
2015-04-03 02:43:50 +08:00
|
|
|
|
2015-04-05 01:53:27 +08:00
|
|
|
if (m_dropPreviewTabs) {
|
2015-04-03 02:43:50 +08:00
|
|
|
m_dropPreviewTabs->removeDropViewPreview();
|
2015-04-05 01:53:27 +08:00
|
|
|
m_dropPreviewTabs = nullptr;
|
|
|
|
}
|
2015-03-28 06:42:18 +08:00
|
|
|
}
|
|
|
|
|
2015-04-21 00:49:25 +08:00
|
|
|
DropViewAtResult Workspace::dropViewAt(const gfx::Point& pos, WorkspaceView* view, bool clone)
|
2015-03-28 05:14:00 +08:00
|
|
|
{
|
2015-04-05 01:53:27 +08:00
|
|
|
WorkspaceTabs* tabs = getTabsAt(pos);
|
|
|
|
WorkspacePanel* panel = getPanelAt(pos);
|
|
|
|
|
2015-04-05 03:45:58 +08:00
|
|
|
if (panel) {
|
|
|
|
// Create new panel
|
2015-04-21 00:49:25 +08:00
|
|
|
return panel->dropViewAt(pos, getViewPanel(view), view, clone);
|
2015-04-05 03:45:58 +08:00
|
|
|
}
|
2015-04-05 01:53:27 +08:00
|
|
|
else if (tabs && tabs != getViewPanel(view)->tabs()) {
|
2015-04-05 03:45:58 +08:00
|
|
|
// Dock tab in other tabs
|
2015-04-05 01:53:27 +08:00
|
|
|
WorkspacePanel* dropPanel = tabs->panel();
|
2015-04-03 02:43:50 +08:00
|
|
|
ASSERT(dropPanel);
|
|
|
|
|
2015-04-05 01:53:27 +08:00
|
|
|
int pos = tabs->getDropTabIndex();
|
2015-04-21 00:49:25 +08:00
|
|
|
DropViewAtResult result;
|
|
|
|
|
|
|
|
if (clone) {
|
|
|
|
view = view->cloneWorkspaceView();
|
|
|
|
result = DropViewAtResult::CLONED_VIEW;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
removeView(view);
|
|
|
|
result = DropViewAtResult::MOVED_TO_OTHER_PANEL;
|
|
|
|
}
|
2015-04-03 02:43:50 +08:00
|
|
|
|
2015-04-05 03:45:58 +08:00
|
|
|
addViewToPanel(dropPanel, view, true, pos);
|
2015-04-21 00:49:25 +08:00
|
|
|
return result;
|
2015-04-03 02:43:50 +08:00
|
|
|
}
|
|
|
|
else
|
2015-04-21 00:49:25 +08:00
|
|
|
return DropViewAtResult::NOTHING;
|
2015-04-03 02:43:50 +08:00
|
|
|
}
|
|
|
|
|
2015-04-05 03:45:58 +08:00
|
|
|
void Workspace::addViewToPanel(WorkspacePanel* panel, WorkspaceView* view, bool from_drop, int pos)
|
2015-04-03 02:43:50 +08:00
|
|
|
{
|
2015-04-05 03:45:58 +08:00
|
|
|
panel->addView(view, from_drop, pos);
|
2015-04-01 04:31:45 +08:00
|
|
|
|
2015-04-03 02:43:50 +08:00
|
|
|
m_activePanel = panel;
|
|
|
|
m_views.push_back(view);
|
|
|
|
|
|
|
|
setActiveView(view);
|
2015-03-28 05:14:00 +08:00
|
|
|
}
|
|
|
|
|
2015-04-01 04:31:45 +08:00
|
|
|
WorkspacePanel* Workspace::getViewPanel(WorkspaceView* view)
|
2015-03-28 05:45:36 +08:00
|
|
|
{
|
2015-04-01 04:31:45 +08:00
|
|
|
Widget* widget = view->getContentWidget();
|
|
|
|
while (widget) {
|
|
|
|
if (widget->getType() == WorkspacePanel::Type())
|
|
|
|
return static_cast<WorkspacePanel*>(widget);
|
2015-03-28 05:45:36 +08:00
|
|
|
|
2015-04-01 04:31:45 +08:00
|
|
|
widget = widget->getParent();
|
|
|
|
}
|
|
|
|
return nullptr;
|
2015-03-28 05:45:36 +08:00
|
|
|
}
|
|
|
|
|
2015-04-01 04:31:45 +08:00
|
|
|
WorkspacePanel* Workspace::getPanelAt(const gfx::Point& pos)
|
2015-03-28 05:45:36 +08:00
|
|
|
{
|
2015-04-01 04:31:45 +08:00
|
|
|
Widget* widget = getManager()->pick(pos);
|
|
|
|
while (widget) {
|
|
|
|
if (widget->getType() == WorkspacePanel::Type())
|
|
|
|
return static_cast<WorkspacePanel*>(widget);
|
|
|
|
|
|
|
|
widget = widget->getParent();
|
|
|
|
}
|
|
|
|
return nullptr;
|
2015-03-28 05:45:36 +08:00
|
|
|
}
|
|
|
|
|
2015-04-03 02:43:50 +08:00
|
|
|
WorkspaceTabs* Workspace::getTabsAt(const gfx::Point& pos)
|
|
|
|
{
|
|
|
|
Widget* widget = getManager()->pick(pos);
|
|
|
|
while (widget) {
|
|
|
|
if (widget->getType() == Tabs::Type())
|
|
|
|
return static_cast<WorkspaceTabs*>(widget);
|
|
|
|
|
|
|
|
widget = widget->getParent();
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2013-08-06 08:20:19 +08:00
|
|
|
} // namespace app
|