aseprite/src/app/commands/cmd_new_frame.cpp

218 lines
6.0 KiB
C++
Raw Normal View History

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.
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include "app/app.h"
#include "app/color.h"
#include "app/commands/command.h"
#include "app/commands/params.h"
#include "app/console.h"
#include "app/context_access.h"
#include "app/document_api.h"
#include "app/modules/gui.h"
#include "app/transaction.h"
#include "app/ui/document_view.h"
#include "app/ui/editor/editor.h"
#include "app/ui/main_window.h"
#include "app/ui/status_bar.h"
#include "app/ui/timeline.h"
#include "app/ui_context.h"
#include "doc/cel.h"
#include "doc/image.h"
#include "doc/layer.h"
#include "doc/sprite.h"
#include "ui/ui.h"
#include <stdexcept>
namespace app {
2015-02-12 23:16:25 +08:00
class NewFrameCommand : public Command {
public:
enum class Content {
DUPLICATE_FRAME,
NEW_EMPTY_FRAME,
DUPLICATE_CELS,
DUPLICATE_CELS_BLOCK,
};
NewFrameCommand();
Command* clone() const override { return new NewFrameCommand(*this); }
protected:
void onLoadParams(const Params& params) override;
bool onEnabled(Context* context) override;
void onExecute(Context* context) override;
std::string onGetFriendlyName() const override;
private:
Content m_content;
};
NewFrameCommand::NewFrameCommand()
: Command("NewFrame",
"New Frame",
CmdRecordableFlag)
{
}
void NewFrameCommand::onLoadParams(const Params& params)
{
m_content = Content::DUPLICATE_FRAME;
std::string content = params.get("content");
if (content == "current" ||
content == "frame")
m_content = Content::DUPLICATE_FRAME;
else if (content == "empty")
m_content = Content::NEW_EMPTY_FRAME;
else if (content == "cel")
m_content = Content::DUPLICATE_CELS;
else if (content == "celblock")
m_content = Content::DUPLICATE_CELS_BLOCK;
}
bool NewFrameCommand::onEnabled(Context* context)
{
return context->checkFlags(ContextFlags::ActiveDocumentIsWritable |
ContextFlags::HasActiveSprite);
}
void NewFrameCommand::onExecute(Context* context)
{
ContextWriter writer(context);
Document* document(writer.document());
Sprite* sprite(writer.sprite());
{
Transaction transaction(writer.context(), "New Frame");
DocumentApi api = document->getApi(transaction);
switch (m_content) {
case Content::DUPLICATE_FRAME:
api.addFrame(sprite, writer.frame()+1);
break;
case Content::NEW_EMPTY_FRAME:
api.addEmptyFrame(sprite, writer.frame()+1);
break;
case Content::DUPLICATE_CELS:
case Content::DUPLICATE_CELS_BLOCK: {
// TODO the range of selected frames should be in doc::Site.
Timeline* timeline = App::instance()->getMainWindow()->getTimeline();
Timeline::Range range = timeline->range();
if (range.enabled()) {
std::map<CelData*, Cel*> relatedCels;
timeline->prepareToMoveRange();
LayerIndex layerBegin = range.layerBegin();
LayerIndex layerEnd = range.layerEnd();
if (range.type() == DocumentRange::kFrames) {
layerBegin = writer.sprite()->firstLayer();
layerEnd = writer.sprite()->lastLayer();
}
for (LayerIndex layer = layerBegin; layer <= layerEnd; ++layer) {
Layer* layerPtr = writer.sprite()->indexToLayer(layer);
if (layerPtr->isImage()) {
for (frame_t frame = range.frameEnd(); frame >= range.frameBegin(); --frame) {
frame_t srcFrame = frame;
frame_t dstFrame = frame+range.frames();
bool continuous;
CelData* srcCelData = nullptr;
if (m_content == Content::DUPLICATE_CELS_BLOCK) {
continuous = false;
Cel* srcCel = static_cast<LayerImage*>(layerPtr)->cel(srcFrame);
if (srcCel) {
srcCelData = srcCel->data();
auto it = relatedCels.find(srcCelData);
if (it != relatedCels.end()) {
srcFrame = it->second->frame();
continuous = true;
}
}
}
else
continuous = layerPtr->isContinuous();
api.copyCel(
static_cast<LayerImage*>(layerPtr), srcFrame,
static_cast<LayerImage*>(layerPtr), dstFrame, continuous);
if (srcCelData && !relatedCels[srcCelData])
relatedCels[srcCelData] = layerPtr->cel(dstFrame);
}
}
}
range.displace(0, range.frames());
timeline->moveRange(range);
}
else {
api.copyCel(
static_cast<LayerImage*>(writer.layer()), writer.frame(),
static_cast<LayerImage*>(writer.layer()), writer.frame()+1);
// TODO should we use DocumentObserver?
if (UIContext::instance() == context) {
if (DocumentView* view = UIContext::instance()->activeView())
view->getEditor()->setFrame(writer.frame()+1);
}
}
break;
}
}
transaction.commit();
}
update_screen_for_document(document);
StatusBar::instance()
->showTip(1000, "New frame %d/%d",
(int)context->activeSite().frame()+1,
(int)sprite->totalFrames());
App::instance()->getMainWindow()->popTimeline();
}
std::string NewFrameCommand::onGetFriendlyName() const
{
std::string text = "New Frame";
switch (m_content) {
case Content::DUPLICATE_FRAME:
text = "New Frame";
break;
case Content::NEW_EMPTY_FRAME:
text = "New Empty Frame";
break;
case Content::DUPLICATE_CELS:
text = "Duplicate Linked Cels";
break;
case Content::DUPLICATE_CELS_BLOCK:
text = "Duplicate Cels";
break;
}
return text;
}
Command* CommandFactory::createNewFrameCommand()
{
return new NewFrameCommand;
}
} // namespace app