aseprite/src/app/ui/doc_view.cpp

601 lines
15 KiB
C++
Raw Normal View History

2015-02-12 23:16:25 +08:00
// Aseprite
// Copyright (C) 2018-2020 Igara Studio S.A.
// Copyright (C) 2001-2018 David Capello
2015-02-12 23:16:25 +08:00
//
2016-08-27 04:02:58 +08:00
// This program is distributed under the terms of
// the End-User License Agreement for Aseprite.
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
2018-07-15 10:24:49 +08:00
#include "app/ui/doc_view.h"
#include "app/app.h"
#include "app/app_menus.h"
#include "app/cmd/clear_mask.h"
#include "app/cmd/deselect_mask.h"
#include "app/cmd/trim_cel.h"
#include "app/commands/commands.h"
#include "app/console.h"
#include "app/context_access.h"
#include "app/doc_access.h"
#include "app/doc_event.h"
#include "app/i18n/strings.h"
#include "app/modules/editors.h"
#include "app/modules/palettes.h"
#include "app/pref/preferences.h"
#include "app/tx.h"
#include "app/ui/editor/editor.h"
#include "app/ui/editor/editor_customization_delegate.h"
#include "app/ui/editor/editor_view.h"
#include "app/ui/keyboard_shortcuts.h"
#include "app/ui/main_window.h"
#include "app/ui/status_bar.h"
#include "app/ui/timeline/timeline.h"
#include "app/ui/workspace.h"
#include "app/ui_context.h"
#include "app/util/clipboard.h"
#include "app/util/range_utils.h"
#include "base/fs.h"
#include "doc/layer.h"
#include "doc/sprite.h"
#include "fmt/format.h"
#include "ui/accelerator.h"
#include "ui/alert.h"
#include "ui/menu.h"
#include "ui/message.h"
#include "ui/system.h"
#include "ui/view.h"
#include <typeinfo>
namespace app {
using namespace ui;
class AppEditor : public Editor,
public EditorObserver,
public EditorCustomizationDelegate {
public:
2018-07-07 22:54:44 +08:00
AppEditor(Doc* document,
2018-07-15 10:24:49 +08:00
DocViewPreviewDelegate* previewDelegate)
: Editor(document)
, m_previewDelegate(previewDelegate) {
add_observer(this);
setCustomizationDelegate(this);
}
~AppEditor() {
remove_observer(this);
setCustomizationDelegate(NULL);
}
// EditorObserver implementation
void dispose() override {
m_previewDelegate->onDisposeOtherEditor(this);
}
void onScrollChanged(Editor* editor) override {
m_previewDelegate->onScrollOtherEditor(this);
if (isActive())
StatusBar::instance()->updateFromEditor(this);
}
void onAfterFrameChanged(Editor* editor) override {
m_previewDelegate->onPreviewOtherEditor(this);
if (isActive())
set_current_palette(editor->sprite()->palette(editor->frame()), false);
}
void onAfterLayerChanged(Editor* editor) override {
m_previewDelegate->onPreviewOtherEditor(this);
}
// EditorCustomizationDelegate implementation
tools::Tool* getQuickTool(tools::Tool* currentTool) override {
return KeyboardShortcuts::instance()
->getCurrentQuicktool(currentTool);
}
KeyAction getPressedKeyAction(KeyContext context) override {
return KeyboardShortcuts::instance()->getCurrentActionModifiers(context);
}
2019-10-02 01:55:08 +08:00
TagProvider* getTagProvider() override {
return App::instance()->mainWindow()->getTimeline();
}
protected:
bool onProcessMessage(Message* msg) override {
switch (msg->type()) {
case kKeyDownMessage:
case kKeyUpMessage:
if (static_cast<KeyMessage*>(msg)->repeat() == 0) {
KeyboardShortcuts* keys = KeyboardShortcuts::instance();
KeyPtr lmb = keys->action(KeyAction::LeftMouseButton);
KeyPtr rmb = keys->action(KeyAction::RightMouseButton);
// Convert action keys into mouse messages.
if (lmb->isPressed(msg, *keys) ||
rmb->isPressed(msg, *keys)) {
MouseMessage mouseMsg(
(msg->type() == kKeyDownMessage ? kMouseDownMessage: kMouseUpMessage),
PointerType::Unknown,
(lmb->isPressed(msg, *keys) ? kButtonLeft: kButtonRight),
msg->modifiers(),
2014-11-26 09:10:28 +08:00
ui::get_mouse_position());
sendMessage(&mouseMsg);
return true;
}
}
break;
}
try {
return Editor::onProcessMessage(msg);
}
catch (const std::exception& ex) {
EditorState* state = getState().get();
Console console;
Console::showException(ex);
console.printf("\nInternal details:\n"
"- Message type: %d\n"
"- Editor state: %s\n",
msg->type(),
state ? typeid(*state).name(): "None");
return false;
}
}
private:
2018-07-15 10:24:49 +08:00
DocViewPreviewDelegate* m_previewDelegate;
};
class PreviewEditor : public Editor,
public EditorCustomizationDelegate {
public:
2018-07-07 22:54:44 +08:00
PreviewEditor(Doc* document)
2017-04-15 02:28:46 +08:00
: Editor(document, Editor::kShowOutside) { // Don't show grid/mask in preview preview
setCustomizationDelegate(this);
}
2017-04-15 02:28:46 +08:00
~PreviewEditor() {
// As we are destroying this instance, we have to remove it as the
// customization delegate. Editor::~Editor() will call
// setCustomizationDelegate(nullptr) too which triggers a
// EditorCustomizationDelegate::dispose() if the customization
// isn't nullptr.
setCustomizationDelegate(nullptr);
}
// EditorCustomizationDelegate implementation
void dispose() override {
// Do nothing
}
tools::Tool* getQuickTool(tools::Tool* currentTool) override {
return nullptr;
}
KeyAction getPressedKeyAction(KeyContext context) override {
return KeyAction::None;
}
2019-10-02 01:55:08 +08:00
TagProvider* getTagProvider() override {
return App::instance()->mainWindow()->getTimeline();
}
};
2018-07-15 10:24:49 +08:00
DocView::DocView(Doc* document, Type type,
DocViewPreviewDelegate* previewDelegate)
: Box(VERTICAL)
, m_type(type)
, m_document(document)
, m_view(new EditorView(type == Normal ? EditorView::CurrentEditorMode:
EditorView::AlwaysSelected))
, m_previewDelegate(previewDelegate)
, m_editor((type == Normal ?
(Editor*)new AppEditor(document, previewDelegate):
(Editor*)new PreviewEditor(document)))
{
addChild(m_view);
m_view->attachToView(m_editor);
m_view->setExpansive(true);
2018-07-15 10:24:49 +08:00
m_editor->setDocView(this);
m_document->add_observer(this);
}
2018-07-15 10:24:49 +08:00
DocView::~DocView()
{
m_document->remove_observer(this);
delete m_editor;
}
2018-07-15 10:24:49 +08:00
void DocView::getSite(Site* site) const
{
m_editor->getSite(site);
}
2018-07-15 10:24:49 +08:00
std::string DocView::getTabText()
{
2015-02-20 00:13:25 +08:00
return m_document->name();
}
2018-07-15 10:24:49 +08:00
TabIcon DocView::getTabIcon()
2015-02-20 08:44:22 +08:00
{
return TabIcon::NONE;
}
2018-07-15 10:24:49 +08:00
WorkspaceView* DocView::cloneWorkspaceView()
{
2018-07-15 10:24:49 +08:00
return new DocView(m_document, Normal, m_previewDelegate);
}
2018-07-15 10:24:49 +08:00
void DocView::onWorkspaceViewSelected()
{
StatusBar::instance()->showDefaultText(m_document);
}
2018-07-15 10:24:49 +08:00
void DocView::onClonedFrom(WorkspaceView* from)
{
Editor* newEditor = this->editor();
2018-07-15 10:24:49 +08:00
Editor* srcEditor = static_cast<DocView*>(from)->editor();
newEditor->setLayer(srcEditor->layer());
newEditor->setFrame(srcEditor->frame());
newEditor->setZoom(srcEditor->zoom());
View::getView(newEditor)
->setViewScroll(View::getView(srcEditor)->viewScroll());
}
2018-07-15 10:24:49 +08:00
bool DocView::onCloseView(Workspace* workspace, bool quitting)
{
if (m_editor->isMovingPixels())
m_editor->dropMovingPixels();
// If there is another view for this document, just close the view.
for (auto view : *workspace) {
2018-07-15 10:24:49 +08:00
DocView* docView = dynamic_cast<DocView*>(view);
if (docView && docView != this &&
docView->document() == document()) {
workspace->removeView(this);
delete this;
return true;
}
}
2015-02-23 21:52:04 +08:00
UIContext* ctx = UIContext::instance();
bool save_it;
bool try_again = true;
while (try_again) {
// This flag indicates if we have to sabe the sprite before to destroy it
save_it = false;
{
// see if the sprite has changes
while (m_document->isModified()) {
// ask what want to do the user with the changes in the sprite
int ret = Alert::show(
fmt::format(
Strings::alerts_save_sprite_changes(),
m_document->name(),
(quitting ? Strings::alerts_save_sprite_changes_quitting():
Strings::alerts_save_sprite_changes_closing())));
if (ret == 1) {
// "save": save the changes
save_it = true;
break;
}
else if (ret != 2) {
// "cancel" or "ESC" */
return false; // we back doing nothing
}
else {
// "discard"
break;
}
}
}
// Does we need to save the sprite?
if (save_it) {
2015-02-23 21:52:04 +08:00
ctx->setActiveView(this);
ctx->updateFlags();
Command* save_command =
Commands::instance()->byId(CommandId::SaveFile());
ctx->executeCommand(save_command);
try_again = true;
}
else
try_again = false;
}
try {
// Destroy the sprite (locking it as writer)
DocDestroyer destroyer(
static_cast<app::Context*>(m_document->context()), m_document, 500);
StatusBar::instance()
->setStatusText(0, "Sprite '%s' closed.",
m_document->name().c_str());
2019-05-28 00:51:41 +08:00
// Just close the document (so we can reopen it with
// ReopenClosedFile command).
destroyer.closeDocument();
// At this point the view is already destroyed
return true;
}
catch (const LockedDocException& ex) {
Console::showException(ex);
return false;
}
}
2018-07-15 10:24:49 +08:00
void DocView::onTabPopup(Workspace* workspace)
{
Menu* menu = AppMenus::instance()->getDocumentTabPopupMenu();
if (!menu)
return;
2015-02-23 21:52:04 +08:00
UIContext* ctx = UIContext::instance();
ctx->setActiveView(this);
ctx->updateFlags();
menu->showPopup(ui::get_mouse_position());
}
2018-07-15 10:24:49 +08:00
bool DocView::onProcessMessage(Message* msg)
{
switch (msg->type()) {
case kFocusEnterMessage:
if (msg->recipient() != m_editor)
m_editor->requestFocus();
break;
}
return Box::onProcessMessage(msg);
}
2018-07-15 10:24:49 +08:00
void DocView::onGeneralUpdate(DocEvent& ev)
{
if (m_editor->isVisible())
m_editor->updateEditor(true);
}
2018-07-15 10:24:49 +08:00
void DocView::onSpritePixelsModified(DocEvent& ev)
{
if (m_editor->isVisible() &&
m_editor->frame() == ev.frame())
m_editor->drawSpriteClipped(ev.region());
}
2018-07-15 10:24:49 +08:00
void DocView::onLayerMergedDown(DocEvent& ev)
{
m_editor->setLayer(ev.targetLayer());
}
2018-07-15 10:24:49 +08:00
void DocView::onAddLayer(DocEvent& ev)
{
if (current_editor == m_editor) {
ASSERT(ev.layer() != NULL);
m_editor->setLayer(ev.layer());
}
}
2018-07-15 10:24:49 +08:00
void DocView::onAddFrame(DocEvent& ev)
{
if (current_editor == m_editor)
m_editor->setFrame(ev.frame());
else if (m_editor->frame() > ev.frame())
m_editor->setFrame(m_editor->frame()+1);
}
2018-07-15 10:24:49 +08:00
void DocView::onRemoveFrame(DocEvent& ev)
{
// Adjust current frame of all editors that are in a frame more
// advanced that the removed one.
if (m_editor->frame() > ev.frame()) {
m_editor->setFrame(m_editor->frame()-1);
}
// If the editor was in the previous "last frame" (current value of
// totalFrames()), we've to adjust it to the new last frame
// (lastFrame())
else if (m_editor->frame() >= m_editor->sprite()->totalFrames()) {
m_editor->setFrame(m_editor->sprite()->lastFrame());
}
}
2018-07-15 10:24:49 +08:00
void DocView::onAddCel(DocEvent& ev)
{
UIContext::instance()->notifyActiveSiteChanged();
}
2018-07-15 10:24:49 +08:00
void DocView::onRemoveCel(DocEvent& ev)
{
// This can happen when we apply a filter that clear the whole cel
// and then the cel is removed in a background/job
// thread. (e.g. applying a convolution matrix)
if (!ui::is_ui_thread())
return;
UIContext::instance()->notifyActiveSiteChanged();
}
2018-07-15 10:24:49 +08:00
void DocView::onTotalFramesChanged(DocEvent& ev)
{
if (m_editor->frame() >= m_editor->sprite()->totalFrames()) {
m_editor->setFrame(m_editor->sprite()->lastFrame());
}
}
2018-07-15 10:24:49 +08:00
void DocView::onLayerRestacked(DocEvent& ev)
{
m_editor->invalidate();
}
2018-07-15 10:24:49 +08:00
void DocView::onNewInputPriority(InputChainElement* element,
const ui::Message* msg)
{
// Do nothing
}
2018-07-15 10:24:49 +08:00
bool DocView::onCanCut(Context* ctx)
{
if (ctx->checkFlags(ContextFlags::ActiveDocumentIsWritable |
ContextFlags::ActiveLayerIsVisible |
ContextFlags::ActiveLayerIsEditable |
ContextFlags::HasVisibleMask |
ContextFlags::HasActiveImage)
&& !ctx->checkFlags(ContextFlags::ActiveLayerIsReference))
return true;
else if (m_editor->isMovingPixels())
return true;
else
return false;
}
2018-07-15 10:24:49 +08:00
bool DocView::onCanCopy(Context* ctx)
{
if (ctx->checkFlags(ContextFlags::ActiveDocumentIsWritable |
ContextFlags::ActiveLayerIsVisible |
ContextFlags::HasVisibleMask |
ContextFlags::HasActiveImage)
&& !ctx->checkFlags(ContextFlags::ActiveLayerIsReference))
return true;
else if (m_editor->isMovingPixels())
return true;
else
return false;
}
2018-07-15 10:24:49 +08:00
bool DocView::onCanPaste(Context* ctx)
{
return
(clipboard::get_current_format() == clipboard::ClipboardImage
&& ctx->checkFlags(ContextFlags::ActiveDocumentIsWritable |
ContextFlags::ActiveLayerIsVisible |
ContextFlags::ActiveLayerIsEditable |
ContextFlags::ActiveLayerIsImage)
&& !ctx->checkFlags(ContextFlags::ActiveLayerIsReference));
}
2018-07-15 10:24:49 +08:00
bool DocView::onCanClear(Context* ctx)
{
if (ctx->checkFlags(ContextFlags::ActiveDocumentIsWritable |
ContextFlags::ActiveLayerIsVisible |
ContextFlags::ActiveLayerIsEditable |
ContextFlags::ActiveLayerIsImage)
&& !ctx->checkFlags(ContextFlags::ActiveLayerIsReference)) {
return true;
}
else if (m_editor->isMovingPixels()) {
return true;
}
else
return false;
}
2018-07-15 10:24:49 +08:00
bool DocView::onCut(Context* ctx)
{
ContextWriter writer(ctx);
clipboard::cut(writer);
return true;
}
2018-07-15 10:24:49 +08:00
bool DocView::onCopy(Context* ctx)
{
const ContextReader reader(ctx);
if (reader.site()->document() &&
2018-07-07 22:54:44 +08:00
static_cast<const Doc*>(reader.site()->document())->isMaskVisible() &&
reader.site()->image()) {
clipboard::copy(reader);
return true;
}
else
return false;
}
2018-07-15 10:24:49 +08:00
bool DocView::onPaste(Context* ctx)
{
if (clipboard::get_current_format() == clipboard::ClipboardImage) {
clipboard::paste(ctx, true);
return true;
}
else
return false;
}
2018-07-15 10:24:49 +08:00
bool DocView::onClear(Context* ctx)
{
// First we check if there is a selected slice, so we'll delete
// those slices.
Site site = ctx->activeSite();
if (!site.selectedSlices().empty()) {
Command* removeSlices = Commands::instance()->byId(CommandId::RemoveSlice());
ctx->executeCommand(removeSlices);
return true;
}
// In other case we delete the mask or the cel.
ContextWriter writer(ctx);
Doc* document = site.document();
bool visibleMask = document->isMaskVisible();
CelList cels;
if (site.range().enabled()) {
cels = get_unlocked_unique_cels(site.sprite(), site.range());
}
else if (site.cel()) {
cels.push_back(site.cel());
}
if (cels.empty()) // No cels to modify
return false;
{
Tx tx(writer.context(), "Clear");
const bool deselectMask =
(visibleMask &&
!Preferences::instance().selection.keepSelectionAfterClear());
clipboard::clear_mask_from_cels(
tx, document, cels,
deselectMask);
tx.commit();
}
if (visibleMask)
document->generateMaskBoundaries();
document->notifyGeneralUpdate();
return true;
}
2018-07-15 10:24:49 +08:00
void DocView::onCancel(Context* ctx)
{
if (m_editor)
m_editor->cancelSelections();
// Deselect mask
if (ctx->checkFlags(ContextFlags::ActiveDocumentIsWritable |
ContextFlags::HasVisibleMask)) {
Command* deselectMask = Commands::instance()->byId(CommandId::DeselectMask());
ctx->executeCommand(deselectMask);
}
}
} // namespace app