aseprite/src/app/commands/cmd_cel_properties.cpp

330 lines
8.4 KiB
C++
Raw Normal View History

2015-02-12 23:16:25 +08:00
// Aseprite
// Copyright (C) 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
#include "app/app.h"
2015-08-21 19:35:29 +08:00
#include "app/cmd/set_cel_opacity.h"
2015-12-11 05:34:25 +08:00
#include "app/cmd/set_user_data.h"
#include "app/commands/command.h"
2015-08-21 19:35:29 +08:00
#include "app/console.h"
#include "app/context_access.h"
#include "app/doc_event.h"
2018-07-07 21:07:21 +08:00
#include "app/doc_range.h"
#include "app/modules/gui.h"
#include "app/tx.h"
#include "app/ui/timeline/timeline.h"
2020-09-11 02:49:08 +08:00
#include "app/ui/user_data_view.h"
2015-08-21 19:35:29 +08:00
#include "app/ui_context.h"
#include "base/mem_utils.h"
2015-08-21 19:35:29 +08:00
#include "base/scoped_value.h"
#include "doc/cel.h"
#include "doc/cels_range.h"
#include "doc/image.h"
#include "doc/layer.h"
#include "doc/sprite.h"
#include "ui/ui.h"
#include "cel_properties.xml.h"
namespace app {
using namespace ui;
2015-08-21 19:35:29 +08:00
class CelPropertiesWindow;
static CelPropertiesWindow* g_window = nullptr;
class CelPropertiesWindow : public app::gen::CelProperties,
public ContextObserver,
public DocObserver {
public:
2015-08-21 19:35:29 +08:00
CelPropertiesWindow()
: m_timer(250, this)
, m_document(nullptr)
2015-08-21 19:35:29 +08:00
, m_cel(nullptr)
, m_selfUpdate(false)
2020-09-11 02:49:08 +08:00
, m_userDataView(new gen::UserData(), &Preferences::instance().cels.userDataVisibility)
{
opacity()->Change.connect([this]{ onStartTimer(); });
2020-09-11 02:49:08 +08:00
userData()->Click.connect([this]{ onToggleUserData(); });
m_timer.Tick.connect([this]{ onCommitChange(); });
2015-08-21 19:35:29 +08:00
2020-09-11 02:49:08 +08:00
m_userDataView.entry()->Change.connect([this]{ onStartTimer(); });
m_userDataView.color()->Change.connect([this]{ onStartTimer(); });
2015-08-21 19:35:29 +08:00
remapWindow();
centerWindow();
load_window_pos(this, "CelProperties");
UIContext::instance()->add_observer(this);
2015-08-21 19:35:29 +08:00
}
~CelPropertiesWindow() {
UIContext::instance()->remove_observer(this);
2015-08-21 19:35:29 +08:00
}
2018-07-07 22:54:44 +08:00
void setCel(Doc* doc, Cel* cel) {
if (m_document) {
m_document->remove_observer(this);
m_document = nullptr;
2015-08-21 19:35:29 +08:00
m_cel = nullptr;
}
m_timer.stop();
m_document = doc;
m_cel = cel;
m_range = App::instance()->timeline()->range();
2015-08-21 19:35:29 +08:00
if (m_document)
m_document->add_observer(this);
2015-08-21 19:35:29 +08:00
2020-09-11 02:49:08 +08:00
if (countCels() > 0 && m_cel) {
ui::Grid* mainGrid = g_window->propertiesGrid();
m_userDataView.configureAndSet(m_cel->data()->userData(), mainGrid);
}
else if (!m_cel)
m_userDataView.setVisible(false, false);
g_window->remapWindow();
manager()->invalidate();
updateFromCel();
}
2015-08-21 19:35:29 +08:00
private:
int opacityValue() const {
return opacity()->getValue();
}
int countCels(int* backgroundCount = nullptr) const {
if (backgroundCount)
*backgroundCount = 0;
if (!m_document)
return 0;
else if (m_cel &&
(!m_range.enabled() ||
(m_range.frames() == 1 &&
m_range.layers() == 1))) {
if (backgroundCount && m_cel->layer()->isBackground())
*backgroundCount = 1;
return 1;
}
else if (m_range.enabled()) {
Sprite* sprite = m_document->sprite();
int count = 0;
for (Cel* cel : sprite->uniqueCels(m_range.selectedFrames())) {
if (m_range.contains(cel->layer())) {
if (backgroundCount && cel->layer()->isBackground())
++(*backgroundCount);
++count;
}
}
return count;
}
else
return 0;
}
2015-08-21 19:35:29 +08:00
bool onProcessMessage(ui::Message* msg) override {
switch (msg->type()) {
case kKeyDownMessage:
if (opacity()->hasFocus()) {
KeyScancode scancode = static_cast<KeyMessage*>(msg)->scancode();
if (scancode == kKeyEnter ||
scancode == kKeyEsc) {
onCommitChange();
closeWindow(this);
return true;
}
}
break;
2015-08-21 19:35:29 +08:00
case kCloseMessage:
// Save changes before we close the window
2020-09-11 02:49:08 +08:00
if (m_cel)
save_window_pos(this, "CelProperties");
setCel(nullptr, nullptr);
2015-08-21 19:35:29 +08:00
deferDelete();
g_window = nullptr;
break;
}
return Window::onProcessMessage(msg);
}
void onStartTimer() {
2015-08-21 19:35:29 +08:00
if (m_selfUpdate)
return;
m_timer.start();
}
void onCommitChange() {
base::ScopedValue<bool> switchSelf(m_selfUpdate, true, false);
m_timer.stop();
const int newOpacity = opacityValue();
2020-09-11 02:49:08 +08:00
const UserData newUserData= m_userDataView.userData();
const int count = countCels();
2015-08-21 19:35:29 +08:00
if ((count > 1) ||
(count == 1 && m_cel && (newOpacity != m_cel->opacity() ||
2020-09-11 02:49:08 +08:00
newUserData != m_cel->data()->userData()))) {
2015-08-21 19:35:29 +08:00
try {
ContextWriter writer(UIContext::instance());
Tx tx(writer.context(), "Set Cel Properties");
2015-08-21 19:35:29 +08:00
2018-07-07 21:07:21 +08:00
DocRange range;
if (m_range.enabled())
range = m_range;
else {
2018-07-07 21:07:21 +08:00
range.startRange(m_cel->layer(), m_cel->frame(), DocRange::kCels);
range.endRange(m_cel->layer(), m_cel->frame());
}
Sprite* sprite = m_document->sprite();
2020-09-11 02:49:08 +08:00
const bool userDataChanged = (newUserData != m_cel->data()->userData());
for (Cel* cel : sprite->uniqueCels(range.selectedFrames())) {
if (range.contains(cel->layer())) {
if (!cel->layer()->isBackground() && newOpacity != cel->opacity()) {
tx(new cmd::SetCelOpacity(cel, newOpacity));
}
2015-12-11 05:34:25 +08:00
2020-09-11 02:49:08 +08:00
if (userDataChanged &&
newUserData != cel->data()->userData()) {
tx(new cmd::SetUserData(cel->data(), newUserData, m_document));
// Redraw timeline because the cel's user data/color
// might have changed.
App::instance()->timeline()->invalidate();
}
}
}
2015-08-21 19:35:29 +08:00
tx.commit();
2015-08-21 19:35:29 +08:00
}
catch (const std::exception& e) {
Console::showException(e);
}
update_screen_for_document(m_document);
2015-08-21 19:35:29 +08:00
}
}
2020-09-11 02:49:08 +08:00
void onToggleUserData() {
if (m_cel) {
m_userDataView.toggleVisibility();
g_window->remapWindow();
manager()->invalidate();
2015-12-11 05:34:25 +08:00
}
}
2015-08-21 19:35:29 +08:00
// ContextObserver impl
void onActiveSiteChange(const Site& site) override {
onCommitChange();
2015-08-21 19:35:29 +08:00
if (isVisible())
2018-07-07 22:54:44 +08:00
setCel(const_cast<Doc*>(site.document()),
const_cast<Cel*>(site.cel()));
else if (m_document)
setCel(nullptr, nullptr);
2015-08-21 19:35:29 +08:00
}
// DocObserver impl
void onBeforeRemoveCel(DocEvent& ev) override {
if (m_cel == ev.cel())
setCel(m_document, nullptr);
}
void onCelOpacityChange(DocEvent& ev) override {
if (m_cel == ev.cel())
updateFromCel();
2015-08-21 19:35:29 +08:00
}
2020-09-11 02:49:08 +08:00
void onUserDataChange(DocEvent& ev) override {
if (m_cel && m_cel->data() == ev.withUserData())
updateFromCel();
}
void updateFromCel() {
2015-08-21 19:35:29 +08:00
if (m_selfUpdate)
return;
m_timer.stop(); // Cancel current editions (just in case)
base::ScopedValue<bool> switchSelf(m_selfUpdate, true, false);
int bgCount = 0;
int count = countCels(&bgCount);
if (count > 0) {
if (m_cel) {
opacity()->setValue(m_cel->opacity());
2020-09-11 02:49:08 +08:00
color_t c = m_cel->data()->userData().color();
m_userDataView.color()->setColor(Color::fromRgb(rgba_getr(c), rgba_getg(c), rgba_getb(c), rgba_geta(c)));
m_userDataView.entry()->setText(m_cel->data()->userData().text());
}
opacity()->setEnabled(bgCount < count);
}
else {
opacity()->setEnabled(false);
2020-09-11 02:49:08 +08:00
m_userDataView.setVisible(false, false);
}
2015-08-21 19:35:29 +08:00
}
Timer m_timer;
2018-07-07 22:54:44 +08:00
Doc* m_document;
2015-08-21 19:35:29 +08:00
Cel* m_cel;
2018-07-07 21:07:21 +08:00
DocRange m_range;
2015-08-21 19:35:29 +08:00
bool m_selfUpdate;
2020-09-11 02:49:08 +08:00
UserDataView m_userDataView;
};
class CelPropertiesCommand : public Command {
public:
CelPropertiesCommand();
protected:
2015-08-21 19:35:29 +08:00
bool onEnabled(Context* context) override;
void onExecute(Context* context) override;
};
CelPropertiesCommand::CelPropertiesCommand()
: Command(CommandId::CelProperties(), CmdUIOnlyFlag)
{
}
bool CelPropertiesCommand::onEnabled(Context* context)
{
return context->checkFlags(ContextFlags::ActiveDocumentIsWritable |
ContextFlags::ActiveLayerIsImage);
}
void CelPropertiesCommand::onExecute(Context* context)
{
2015-08-21 19:35:29 +08:00
ContextReader reader(context);
2015-08-21 19:35:29 +08:00
if (!g_window)
g_window = new CelPropertiesWindow;
g_window->setCel(reader.document(), reader.cel());
2015-08-21 19:35:29 +08:00
g_window->openWindow();
2015-08-21 19:35:29 +08:00
// Focus layer name
g_window->opacity()->requestFocus();
}
Command* CommandFactory::createCelPropertiesCommand()
{
return new CelPropertiesCommand;
}
} // namespace app