aseprite/src/app/ui/workspace.cpp

418 lines
9.8 KiB
C++

// Aseprite
// Copyright (C) 2018-2024 Igara Studio S.A.
// Copyright (C) 2001-2018 David Capello
//
// This program is distributed under the terms of
// the End-User License Agreement for Aseprite.
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include "app/ui/workspace.h"
#include "app/app.h"
#include "app/ui/input_chain.h"
#include "app/ui/skin/skin_theme.h"
#include "app/ui/workspace_tabs.h"
#include "app/ui/workspace_view.h"
#include "base/remove_from_container.h"
#include "ui/paint_event.h"
#include "ui/resize_event.h"
namespace app {
using namespace app::skin;
using namespace ui;
// static
WidgetType Workspace::Type()
{
static WidgetType type = kGenericWidget;
if (type == kGenericWidget)
type = register_widget_type();
return type;
}
Workspace::Workspace()
: Widget(Workspace::Type())
, m_mainPanel(WorkspacePanel::MAIN_PANEL)
, m_tabs(nullptr)
, m_activePanel(&m_mainPanel)
, m_dropPreviewPanel(nullptr)
, m_dropPreviewTabs(nullptr)
{
enableFlags(IGNORE_MOUSE);
addChild(&m_mainPanel);
InitTheme.connect(
[this]{
auto theme = SkinTheme::get(this);
setBgColor(theme->colors.workspace());
});
initTheme();
}
Workspace::~Workspace()
{
// No views at this point.
ASSERT(m_views.empty());
}
void Workspace::setTabsBar(WorkspaceTabs* tabs)
{
m_tabs = tabs;
m_mainPanel.setTabsBar(tabs);
}
void Workspace::addView(WorkspaceView* view, int pos)
{
addViewToPanel(&m_mainPanel, view, false, pos);
}
void Workspace::removeView(WorkspaceView* view)
{
ASSERT(view);
base::remove_from_container(m_views, view);
WorkspacePanel* panel = getViewPanel(view);
ASSERT(panel);
if (panel)
panel->removeView(view);
view->onAfterRemoveView(this);
}
bool Workspace::closeView(WorkspaceView* view, bool quitting)
{
return view->onCloseView(this, quitting);
}
WorkspaceView* Workspace::activeView()
{
return (m_activePanel ? m_activePanel->activeView(): nullptr);
}
void Workspace::setActiveView(WorkspaceView* view)
{
m_activePanel = getViewPanel(view);
ASSERT(m_activePanel);
if (!m_activePanel)
return;
BeforeViewChanged();
m_activePanel->setActiveView(view);
ActiveViewChanged(); // Fire ActiveViewChanged event
}
void Workspace::setMainPanelAsActive()
{
BeforeViewChanged();
m_activePanel = &m_mainPanel;
removeDropViewPreview();
m_dropPreviewPanel = nullptr;
m_dropPreviewTabs = nullptr;
ActiveViewChanged(); // Fire ActiveViewChanged event
}
bool Workspace::canSelectOtherTab() const
{
return m_activePanel->tabs()->canSelectOtherTab();
}
void Workspace::selectNextTab()
{
m_activePanel->tabs()->selectNextTab();
}
void Workspace::selectPreviousTab()
{
m_activePanel->tabs()->selectPreviousTab();
}
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);
}
void Workspace::updateTabs()
{
WidgetsList children = this->children();
while (!children.empty()) {
Widget* child = children.back();
children.erase(--children.end());
if (child->type() == WorkspacePanel::Type())
static_cast<WorkspacePanel*>(child)->tabs()->updateTabs();
for (auto subchild : child->children())
children.push_back(subchild);
}
}
void Workspace::onPaint(PaintEvent& ev)
{
ev.graphics()->fillRect(bgColor(), clientBounds());
}
void Workspace::onResize(ui::ResizeEvent& ev)
{
setBoundsQuietly(ev.bounds());
gfx::Rect rc = childrenBounds();
for (auto child : children())
child->setBounds(rc);
}
DropViewPreviewResult Workspace::setDropViewPreview(
const gfx::Point& screenPos,
WorkspaceView* view,
WorkspaceTabs* tabs)
{
TabView* tabView = dynamic_cast<TabView*>(view);
WorkspaceTabs* newTabs = nullptr;
WorkspacePanel* panel = getPanelAt(screenPos);
if (!newTabs) {
newTabs = getTabsAt(screenPos);
// Drop preview is only to drop tabs from a different WorkspaceTabs.
if (newTabs == tabs)
newTabs = nullptr;
}
if (m_dropPreviewPanel && m_dropPreviewPanel != panel)
m_dropPreviewPanel->removeDropViewPreview();
if (m_dropPreviewTabs && m_dropPreviewTabs != newTabs)
m_dropPreviewTabs->removeDropViewPreview();
m_dropPreviewPanel = panel;
m_dropPreviewTabs = newTabs;
if (m_dropPreviewPanel)
m_dropPreviewPanel->setDropViewPreview(screenPos, view);
if (m_dropPreviewTabs)
m_dropPreviewTabs->setDropViewPreview(screenPos, tabView);
if (panel)
return DropViewPreviewResult::DROP_IN_PANEL;
else if (newTabs)
return DropViewPreviewResult::DROP_IN_TABS;
else
return DropViewPreviewResult::FLOATING;
}
void Workspace::removeDropViewPreview()
{
if (m_dropPreviewPanel) {
m_dropPreviewPanel->removeDropViewPreview();
m_dropPreviewPanel = nullptr;
}
if (m_dropPreviewTabs) {
m_dropPreviewTabs->removeDropViewPreview();
m_dropPreviewTabs = nullptr;
}
}
DropViewAtResult Workspace::dropViewAt(const gfx::Point& screenPos,
WorkspaceView* view,
const bool clone)
{
WorkspaceTabs* tabs = getTabsAt(screenPos);
WorkspacePanel* panel = getPanelAt(screenPos);
if (panel) {
// Create new panel
return panel->dropViewAt(screenPos, getViewPanel(view), view, clone);
}
else if (tabs && tabs != getViewPanel(view)->tabs()) {
// Dock tab in other tabs
WorkspacePanel* dropPanel = tabs->panel();
ASSERT(dropPanel);
int pos = tabs->getDropTabIndex();
DropViewAtResult result;
WorkspaceView* originalView = view;
if (clone) {
view = view->cloneWorkspaceView();
result = DropViewAtResult::CLONED_VIEW;
}
else {
removeView(view);
result = DropViewAtResult::MOVED_TO_OTHER_PANEL;
}
addViewToPanel(dropPanel, view, true, pos);
if (result == DropViewAtResult::CLONED_VIEW)
view->onClonedFrom(originalView);
return result;
}
else
return DropViewAtResult::NOTHING;
}
void Workspace::addViewToPanel(WorkspacePanel* panel, WorkspaceView* view, bool from_drop, int pos)
{
panel->addView(view, from_drop, pos);
m_activePanel = panel;
m_views.push_back(view);
setActiveView(view);
layout();
}
WorkspacePanel* Workspace::getViewPanel(WorkspaceView* view)
{
Widget* widget = view->getContentWidget();
while (widget) {
if (widget->type() == WorkspacePanel::Type())
return static_cast<WorkspacePanel*>(widget);
widget = widget->parent();
}
return nullptr;
}
WorkspacePanel* Workspace::getPanelAt(const gfx::Point& screenPos)
{
Widget* widget = manager()->pickFromScreenPos(screenPos);
while (widget) {
if (widget->type() == WorkspacePanel::Type())
return static_cast<WorkspacePanel*>(widget);
widget = widget->parent();
}
return nullptr;
}
WorkspaceTabs* Workspace::getTabsAt(const gfx::Point& screenPos)
{
Widget* widget = manager()->pickFromScreenPos(screenPos);
while (widget) {
if (widget->type() == Tabs::Type())
return static_cast<WorkspaceTabs*>(widget);
widget = widget->parent();
}
return nullptr;
}
void Workspace::onNewInputPriority(InputChainElement* newElement,
const ui::Message* msg)
{
WorkspaceView* view = activeView();
InputChainElement* activeElement = (view ? view->onGetInputChainElement(): nullptr);
if (activeElement)
activeElement->onNewInputPriority(newElement, msg);
}
bool Workspace::onCanCut(Context* ctx)
{
WorkspaceView* view = activeView();
InputChainElement* activeElement = (view ? view->onGetInputChainElement(): nullptr);
if (activeElement)
return activeElement->onCanCut(ctx);
else
return false;
}
bool Workspace::onCanCopy(Context* ctx)
{
WorkspaceView* view = activeView();
InputChainElement* activeElement = (view ? view->onGetInputChainElement(): nullptr);
if (activeElement)
return activeElement->onCanCopy(ctx);
else
return false;
}
bool Workspace::onCanPaste(Context* ctx)
{
WorkspaceView* view = activeView();
InputChainElement* activeElement = (view ? view->onGetInputChainElement(): nullptr);
if (activeElement)
return activeElement->onCanPaste(ctx);
else
return false;
}
bool Workspace::onCanClear(Context* ctx)
{
WorkspaceView* view = activeView();
InputChainElement* activeElement = (view ? view->onGetInputChainElement(): nullptr);
if (activeElement)
return activeElement->onCanClear(ctx);
else
return false;
}
bool Workspace::onCut(Context* ctx)
{
WorkspaceView* view = activeView();
InputChainElement* activeElement = (view ? view->onGetInputChainElement(): nullptr);
if (activeElement)
return activeElement->onCut(ctx);
else
return false;
}
bool Workspace::onCopy(Context* ctx)
{
WorkspaceView* view = activeView();
InputChainElement* activeElement = (view ? view->onGetInputChainElement(): nullptr);
if (activeElement)
return activeElement->onCopy(ctx);
else
return false;
}
bool Workspace::onPaste(Context* ctx,
const gfx::Point* position)
{
WorkspaceView* view = activeView();
InputChainElement* activeElement = (view ? view->onGetInputChainElement(): nullptr);
if (activeElement)
return activeElement->onPaste(ctx, position);
else
return false;
}
bool Workspace::onClear(Context* ctx)
{
WorkspaceView* view = activeView();
InputChainElement* activeElement = (view ? view->onGetInputChainElement(): nullptr);
if (activeElement)
return activeElement->onClear(ctx);
else
return false;
}
void Workspace::onCancel(Context* ctx)
{
WorkspaceView* view = activeView();
InputChainElement* activeElement = (view ? view->onGetInputChainElement(): nullptr);
if (activeElement)
activeElement->onCancel(ctx);
}
} // namespace app