#include "tools/strokeselection.h"
// TnzTools includes
#include "tools/imagegrouping.h"
#include "tools/toolhandle.h"
#include "tools/tool.h"
#include "tools/toolutils.h"
// TnzQt includes
#include "toonzqt/selectioncommandids.h"
#include "toonzqt/imageutils.h"
#include "toonzqt/tselectionhandle.h"
#include "toonzqt/strokesdata.h"
#include "toonzqt/rasterimagedata.h"
#include "toonzqt/dvdialog.h"
// TnzLib includes
#include "toonz/tpalettehandle.h"
#include "toonz/palettecontroller.h"
#include "toonz/tobjecthandle.h"
#include "toonz/txshlevelhandle.h"
#include "toonz/tscenehandle.h"
#include "toonz/tcenterlinevectorizer.h"
#include "toonz/stage.h"
#include "toonz/tstageobject.h"
#include "toonz/toonzscene.h"
#include "toonz/sceneproperties.h"
// TnzCore includes
#include "tthreadmessage.h"
#include "tundo.h"
#include "tstroke.h"
#include "tvectorimage.h"
#include "tcolorstyles.h"
#include "tpalette.h"
// Qt includes
#include <QApplication>
#include <QClipboard>
//=============================================================================
namespace
{
void vectorizeToonzImageData(const TVectorImageP &image, const ToonzImageData *tiData,
std::set<int> &indexes, TPalette *palette,
const VectorizerConfiguration &config)
{
if (!tiData)
return;
QApplication::setOverrideCursor(Qt::WaitCursor);
TRasterP ras;
std::vector<TRectD> rects;
std::vector<TStroke> strokes;
std::vector<TStroke> originalStrokes;
TAffine affine;
double dpiX, dpiY;
tiData->getData(ras, dpiX, dpiY, rects, strokes, originalStrokes, affine, image->getPalette());
TRasterCM32P rasCM = ras;
TToonzImageP ti(rasCM, rasCM->getBounds());
VectorizerCore vc;
TVectorImageP vi = vc.vectorize(ti, config, palette);
assert(vi);
vi->setPalette(palette);
TScale sc(dpiX / Stage::inch, dpiY / Stage::inch);
int i;
TRectD selectionBounds;
for (i = 0; i < (int)rects.size(); i++)
selectionBounds += rects[i];
for (i = 0; i < (int)strokes.size(); i++)
selectionBounds += strokes[i].getBBox();
TTranslation tr(selectionBounds.getP00());
for (i = 0; i < (int)vi->getStrokeCount(); i++) {
TStroke *stroke = vi->getStroke(i);
stroke->transform(sc.inv() * affine * tr, true);
}
UINT oldImageSize = image->getStrokeCount();
image->mergeImage(vi, TAffine());
UINT newImageSize = image->getStrokeCount();
indexes.clear();
for (UINT sI = oldImageSize; sI < newImageSize; sI++)
indexes.insert(sI);
QApplication::restoreOverrideCursor();
}
//-----------------------------------------------------------------------------
void copyStrokesWithoutUndo(TVectorImageP image, std::set<int> &indexes)
{
QClipboard *clipboard = QApplication::clipboard();
StrokesData *data = new StrokesData();
data->setImage(image, indexes);
clipboard->setMimeData(data, QClipboard::Clipboard);
}
//-----------------------------------------------------------------------------
bool pasteStrokesWithoutUndo(TVectorImageP image, std::set<int> &outIndexes,
TSceneHandle *sceneHandle, bool insert = true)
{
QMutexLocker lock(image->getMutex());
QClipboard *clipboard = QApplication::clipboard();
const StrokesData *stData = dynamic_cast<const StrokesData *>(clipboard->mimeData());
const ToonzImageData *tiData = dynamic_cast<const ToonzImageData *>(clipboard->mimeData());
const FullColorImageData *fciData = dynamic_cast<const FullColorImageData *>(clipboard->mimeData());
std::set<int> indexes = outIndexes;
if (stData)
stData->getImage(image, indexes, insert);
else if (tiData) {
ToonzScene *scene = sceneHandle->getScene();
assert(scene);
const VectorizerParameters *vParams = scene->getProperties()->getVectorizerParameters();
assert(vParams);
std::auto_ptr<VectorizerConfiguration> config(vParams->getCurrentConfiguration(0.0));
vectorizeToonzImageData(image, tiData, indexes, image->getPalette(), *config);
} else if (fciData) {
DVGui::error(QObject::tr("The copied selection cannot be pasted in the current drawing."));
return false;
} else
return false;
StrokeSelection *selection = dynamic_cast<StrokeSelection *>(TTool::getApplication()->getCurrentSelection()->getSelection());
if (selection)
selection->notifyView();
outIndexes = indexes; //outIndexes is a reference to current selection, so the notifyImageChanged could reset it!
return true;
}
//-----------------------------------------------------------------------------
void deleteStrokesWithoutUndo(TVectorImageP image, std::set<int> &indexes)
{
QMutexLocker lock(image->getMutex());
std::vector<int> indexesV(indexes.begin(), indexes.end());
TRectD bbox;
UINT i = 0;
for (; i < indexesV.size(); i++)
bbox += image->getStroke(indexesV[i])->getBBox();
std::vector<TFilledRegionInf> regions;
ImageUtils::getFillingInformationOverlappingArea(image, regions, bbox);
TVectorImageP other = image->splitImage(indexesV, true);
indexes.clear();
TTool::getApplication()->getCurrentTool()->getTool()->notifyImageChanged();
StrokeSelection *selection = dynamic_cast<StrokeSelection *>(TTool::getApplication()->getCurrentSelection()->getSelection());
if (selection)
selection->notifyView();
}
//-----------------------------------------------------------------------------
void cutStrokesWithoutUndo(TVectorImageP image, std::set<int> &indexes)
{
copyStrokesWithoutUndo(image, indexes);
deleteStrokesWithoutUndo(image, indexes);
}
//=============================================================================
// CopyStrokesUndo
//-----------------------------------------------------------------------------
class CopyStrokesUndo : public TUndo
{
QMimeData *m_oldData;
QMimeData *m_newData;
public:
CopyStrokesUndo(QMimeData *oldData,
QMimeData *newData)
: m_oldData(oldData), m_newData(newData)
{
}
void undo() const
{
QClipboard *clipboard = QApplication::clipboard();
clipboard->setMimeData(cloneData(m_oldData), QClipboard::Clipboard);
}
void redo() const
{
QClipboard *clipboard = QApplication::clipboard();
clipboard->setMimeData(cloneData(m_newData), QClipboard::Clipboard);
}
int getSize() const
{
return sizeof(*this);
}
};
//=============================================================================
// PasteStrokesUndo
//-----------------------------------------------------------------------------
class PasteStrokesUndo : public TUndo
{
TXshSimpleLevelP m_level;
TFrameId m_frameId;
std::set<int> m_indexes;
TPaletteP m_oldPalette;
QMimeData *m_oldData;
TSceneHandle *m_sceneHandle;
public:
PasteStrokesUndo(TXshSimpleLevel *level,
const TFrameId &frameId,
std::set<int> &indexes,
TPaletteP oldPalette,
TSceneHandle *sceneHandle)
: m_level(level), m_frameId(frameId), m_indexes(indexes), m_oldPalette(oldPalette), m_sceneHandle(sceneHandle)
{
QClipboard *clipboard = QApplication::clipboard();
m_oldData = cloneData(clipboard->mimeData());
}
~PasteStrokesUndo()
{
delete m_oldData;
}
void undo() const
{
TVectorImageP image = m_level->getFrame(m_frameId, true);
// Se la selezione corrente e' la stroke selection devo svuotarla,
// altrimenti puo' rimanere selezionato uno stroke che non esiste piu'.
StrokeSelection *selection = dynamic_cast<StrokeSelection *>(TTool::getApplication()->getCurrentSelection()->getSelection());
if (selection)
selection->selectNone();
std::set<int> indexes = m_indexes;
deleteStrokesWithoutUndo(image, indexes);
}
void redo() const
{
TVectorImageP image = m_level->getFrame(m_frameId, true);
std::set<int> indexes = m_indexes;
QClipboard *clipboard = QApplication::clipboard();
QMimeData *data = cloneData(clipboard->mimeData());
clipboard->setMimeData(cloneData(m_oldData), QClipboard::Clipboard);
pasteStrokesWithoutUndo(image, indexes, m_sceneHandle);
TTool::getApplication()->getCurrentTool()->getTool()->notifyImageChanged();
clipboard->setMimeData(data, QClipboard::Clipboard);
}
int getSize() const
{
return sizeof(*this);
}
};
//--------------------------------------------------------------------
class RemoveEndpointsUndo : public TUndo
{
TXshSimpleLevelP m_level;
TFrameId m_frameId;
std::vector<std::pair<int, TStroke *>> m_strokes;
public:
RemoveEndpointsUndo(TXshSimpleLevel *level,
const TFrameId &frameId,
std::vector<std::pair<int, TStroke *>> strokes)
: m_level(level), m_frameId(frameId), m_strokes(strokes)
{
}
~RemoveEndpointsUndo()
{
int i;
for (i = 0; i < (int)m_strokes.size(); i++)
delete m_strokes[i].second;
}
void undo() const
{
TVectorImageP vi = m_level->getFrame(m_frameId, true);
int i;
for (i = 0; i < (int)m_strokes.size(); i++) {
TStroke *newS = new TStroke(*(m_strokes[i].second));
newS->setId(m_strokes[i].second->getId());
vi->restoreEndpoints(m_strokes[i].first, newS);
}
StrokeSelection *selection = dynamic_cast<StrokeSelection *>(TTool::getApplication()->getCurrentSelection()->getSelection());
if (selection)
selection->selectNone();
TTool::getApplication()->getCurrentTool()->getTool()->notifyImageChanged();
}
void redo() const
{
int i;
TVectorImageP vi = m_level->getFrame(m_frameId, true);
for (i = 0; i < (int)m_strokes.size(); i++) {
TStroke *s = vi->removeEndpoints(m_strokes[i].first);
delete s;
//assert(s==m_strokes[i].second);
}
TTool::getApplication()->getCurrentTool()->getTool()->notifyImageChanged();
}
int getSize() const
{
return sizeof(*this);
}
};
//=============================================================================
// DeleteFramesUndo
//-----------------------------------------------------------------------------
class DeleteStrokesUndo : public TUndo
{
protected:
TXshSimpleLevelP m_level;
TFrameId m_frameId;
std::set<int> m_indexes;
QMimeData *m_data;
TSceneHandle *m_sceneHandle;
public:
DeleteStrokesUndo(TXshSimpleLevel *level,
const TFrameId &frameId,
std::set<int> indexes, QMimeData *data,
TSceneHandle *sceneHandle)
: m_level(level), m_frameId(frameId), m_indexes(indexes), m_data(data), m_sceneHandle(sceneHandle)
{
}
~DeleteStrokesUndo()
{
delete m_data;
}
void undo() const
{
QClipboard *clipboard = QApplication::clipboard();
QMimeData *oldData = cloneData(clipboard->mimeData());
clipboard->setMimeData(cloneData(m_data), QClipboard::Clipboard);
std::set<int> indexes = m_indexes;
TVectorImageP image = m_level->getFrame(m_frameId, true);
pasteStrokesWithoutUndo(image, indexes, m_sceneHandle, false);
TTool::getApplication()->getCurrentTool()->getTool()->notifyImageChanged();
clipboard->setMimeData(oldData, QClipboard::Clipboard);
}
void redo() const
{
TVectorImageP image = m_level->getFrame(m_frameId, true);
std::set<int> indexes = m_indexes;
deleteStrokesWithoutUndo(image, indexes);
}
int getSize() const
{
return sizeof(*this);
}
};
//=============================================================================
// CutStrokesUndo
//-----------------------------------------------------------------------------
class CutStrokesUndo : public DeleteStrokesUndo
{
public:
CutStrokesUndo(TXshSimpleLevel *level,
const TFrameId &frameId,
std::set<int> indexes, QMimeData *data,
TSceneHandle *sceneHandle)
: DeleteStrokesUndo(level, frameId, indexes, data, sceneHandle)
{
}
~CutStrokesUndo()
{
}
void redo() const
{
TVectorImageP image = m_level->getFrame(m_frameId, true);
std::set<int> indexes = m_indexes;
cutStrokesWithoutUndo(image, indexes);
}
};
} // namespace
//=============================================================================
//
// StrokeSelection ctor/dtor
//
//-----------------------------------------------------------------------------
StrokeSelection::StrokeSelection()
: m_groupCommand(new TGroupCommand()), m_sceneHandle(), m_updateSelectionBBox(false)
{
m_groupCommand->setSelection(this);
}
//-----------------------------------------------------------------------------
StrokeSelection::~StrokeSelection()
{
}
//-----------------------------------------------------------------------------
StrokeSelection::StrokeSelection(const StrokeSelection &other)
: m_vi(other.m_vi), m_indexes(other.m_indexes), m_groupCommand(new TGroupCommand()), m_sceneHandle(other.m_sceneHandle), m_updateSelectionBBox(other.m_updateSelectionBBox)
{
m_groupCommand->setSelection(this);
}
//-----------------------------------------------------------------------------
StrokeSelection &StrokeSelection::operator=(const StrokeSelection &other)
{
m_vi = other.m_vi;
m_indexes = other.m_indexes;
m_sceneHandle = other.m_sceneHandle;
m_updateSelectionBBox = other.m_updateSelectionBBox;
return *this;
}
//-----------------------------------------------------------------------------
void StrokeSelection::select(int index, bool on)
{
if (on)
m_indexes.insert(index);
else
m_indexes.erase(index);
}
//-----------------------------------------------------------------------------
void StrokeSelection::toggle(int index)
{
std::set<int>::iterator it = m_indexes.find(index);
if (it == m_indexes.end())
m_indexes.insert(index);
else
m_indexes.erase(it);
}
//=============================================================================
//
// removeEndpoints
//
//-----------------------------------------------------------------------------
void StrokeSelection::removeEndpoints()
{
if (!m_vi)
return;
if (m_indexes.empty())
return;
std::vector<std::pair<int, TStroke *>> undoData;
m_vi->findRegions();
set<int>::iterator it = m_indexes.begin();
for (; it != m_indexes.end(); ++it) {
TStroke *s = m_vi->removeEndpoints(*it);
if (s)
undoData.push_back(std::pair<int, TStroke *>(*it, s));
}
TTool *tool = TTool::getApplication()->getCurrentTool()->getTool();
TXshSimpleLevel *level = TTool::getApplication()->getCurrentLevel()->getSimpleLevel();
if (!undoData.empty())
TUndoManager::manager()->add(new RemoveEndpointsUndo(level, tool->getCurrentFid(), undoData));
m_updateSelectionBBox = true;
tool->notifyImageChanged();
m_updateSelectionBBox = false;
}
//=============================================================================
//
// deleteStrokes
//
//-----------------------------------------------------------------------------
void StrokeSelection::deleteStrokes()
{
if (!m_vi)
return;
if (m_indexes.empty())
return;
TTool *tool = TTool::getApplication()->getCurrentTool()->getTool();
if (!tool)
return;
bool isSpline = tool->getApplication()->getCurrentObject()->isSpline();
TUndo *undo;
if (isSpline)
undo = new ToolUtils::UndoPath(tool->getXsheet()->getStageObject(tool->getObjectId())->getSpline());
StrokesData *data = new StrokesData();
data->setImage(m_vi, m_indexes);
std::set<int> oldIndexes = m_indexes;
deleteStrokesWithoutUndo(m_vi, m_indexes);
if (!isSpline) {
TXshSimpleLevel *level = TTool::getApplication()->getCurrentLevel()->getSimpleLevel();
TUndoManager::manager()->add(new DeleteStrokesUndo(level, tool->getCurrentFid(), oldIndexes, data, m_sceneHandle));
} else {
assert(undo);
if (undo)
TUndoManager::manager()->add(undo);
}
}
//=============================================================================
//
// StrokeSelection::copy()
//
//-----------------------------------------------------------------------------
void StrokeSelection::copy()
{
if (m_indexes.empty())
return;
QClipboard *clipboard = QApplication::clipboard();
QMimeData *oldData = cloneData(clipboard->mimeData());
copyStrokesWithoutUndo(m_vi, m_indexes);
QMimeData *newData = cloneData(clipboard->mimeData());
//TUndoManager::manager()->add(new CopyStrokesUndo(oldData, newData));
}
//=============================================================================
//
// StrokeSelection::paste()
//
//-----------------------------------------------------------------------------
void StrokeSelection::paste()
{
TTool *tool = TTool::getApplication()->getCurrentTool()->getTool();
if (!tool)
return;
if (TTool::getApplication()->getCurrentObject()->isSpline()) {
const StrokesData *stData = dynamic_cast<const StrokesData *>(QApplication::clipboard()->mimeData());
if (!stData)
return;
TVectorImageP splineImg = tool->getImage(true);
TVectorImageP img = stData->m_image;
if (!splineImg || !img)
return;
QMutexLocker lock(splineImg->getMutex());
TUndo *undo = new ToolUtils::UndoPath(tool->getXsheet()->getStageObject(tool->getObjectId())->getSpline());
while (splineImg->getStrokeCount() > 0)
splineImg->deleteStroke(0);
TStroke *stroke = img->getStroke(0);
splineImg->addStroke(new TStroke(*stroke), false);
TUndoManager::manager()->add(undo);
tool->notifyImageChanged();
tool->invalidate();
return;
}
TVectorImageP tarImg = tool->touchImage();
if (!tarImg)
return;
TPaletteP palette = tarImg->getPalette();
TPaletteP oldPalette = new TPalette();
if (palette)
oldPalette = palette->clone();
bool isPaste = pasteStrokesWithoutUndo(tarImg, m_indexes, m_sceneHandle);
if (isPaste) {
TXshSimpleLevel *level = TTool::getApplication()->getCurrentLevel()->getSimpleLevel();
TUndoManager::manager()->add(new PasteStrokesUndo(level, tool->getCurrentFid(), m_indexes, oldPalette, m_sceneHandle));
m_updateSelectionBBox = isPaste;
}
tool->notifyImageChanged();
tool->getApplication()->getPaletteController()->getCurrentLevelPalette()->notifyPaletteChanged();
m_updateSelectionBBox = false;
tool->invalidate();
}
//=============================================================================
//
// StrokeSelection::cut()
//
//-----------------------------------------------------------------------------
void StrokeSelection::cut()
{
if (m_indexes.empty())
return;
TTool *tool = TTool::getApplication()->getCurrentTool()->getTool();
if (!tool)
return;
bool isSpline = tool->getApplication()->getCurrentObject()->isSpline();
TUndo *undo;
if (isSpline)
undo = new ToolUtils::UndoPath(tool->getXsheet()->getStageObject(tool->getObjectId())->getSpline());
StrokesData *data = new StrokesData();
data->setImage(m_vi, m_indexes);
std::set<int> oldIndexes = m_indexes;
cutStrokesWithoutUndo(m_vi, m_indexes);
if (!isSpline) {
TXshSimpleLevel *level = tool->getApplication()->getCurrentLevel()->getSimpleLevel();
TUndoManager::manager()->add(new CutStrokesUndo(level, tool->getCurrentFid(), oldIndexes, data, m_sceneHandle));
} else {
assert(undo);
if (undo)
TUndoManager::manager()->add(undo);
}
}
//=============================================================================
//
// enableCommands
//
//-----------------------------------------------------------------------------
void StrokeSelection::enableCommands()
{
enableCommand(this, MI_Clear, &StrokeSelection::deleteStrokes);
enableCommand(this, MI_Cut, &StrokeSelection::cut);
enableCommand(this, MI_Copy, &StrokeSelection::copy);
enableCommand(this, MI_Paste, &StrokeSelection::paste);
enableCommand(m_groupCommand.get(), MI_Group, &TGroupCommand::group);
enableCommand(m_groupCommand.get(), MI_Ungroup, &TGroupCommand::ungroup);
enableCommand(m_groupCommand.get(), MI_BringToFront, &TGroupCommand::front);
enableCommand(m_groupCommand.get(), MI_BringForward, &TGroupCommand::forward);
enableCommand(m_groupCommand.get(), MI_SendBack, &TGroupCommand::back);
enableCommand(m_groupCommand.get(), MI_SendBackward, &TGroupCommand::backward);
enableCommand(m_groupCommand.get(), MI_EnterGroup, &TGroupCommand::enterGroup);
enableCommand(m_groupCommand.get(), MI_ExitGroup, &TGroupCommand::exitGroup);
enableCommand(this, MI_RemoveEndpoints, &StrokeSelection::removeEndpoints);
}
//===================================================================
namespace
{
//===================================================================
class UndoSetStrokeStyle : public TUndo
{
TVectorImageP m_image;
std::vector<int> m_strokeIndexes;
std::vector<int> m_oldStyles;
int m_newStyle;
public:
UndoSetStrokeStyle(TVectorImageP image, int newStyle)
: m_image(image), m_newStyle(newStyle)
{
}
void addStroke(TStroke *stroke)
{
m_strokeIndexes.push_back(m_image->getStrokeIndex(stroke));
m_oldStyles.push_back(stroke->getStyle());
}
void undo() const
{
UINT size = m_strokeIndexes.size();
assert(size == m_oldStyles.size());
for (UINT i = 0; i != size; i++) {
int index = m_strokeIndexes[i];
if (index != -1 && index < (int)m_image->getStrokeCount())
m_image->getStroke(index)->setStyle(m_oldStyles[i]);
}
TTool::getApplication()->getCurrentTool()->getTool()->notifyImageChanged();
}
void redo() const
{
UINT size = m_strokeIndexes.size();
assert(size == m_oldStyles.size());
for (UINT i = 0; i != size; i++) {
int index = m_strokeIndexes[i];
if (index != -1 && index < (int)m_image->getStrokeCount())
m_image->getStroke(index)->setStyle(m_newStyle);
}
TTool::getApplication()->getCurrentTool()->getTool()->notifyImageChanged();
}
int getSize() const
{
return sizeof(*this) +
m_strokeIndexes.capacity() * sizeof(m_strokeIndexes[0]) +
m_oldStyles.capacity() * sizeof(m_oldStyles[0]);
}
};
} // namespace
//=============================================================================
//
// changeColorStyle
//
//-----------------------------------------------------------------------------
void StrokeSelection::changeColorStyle(int styleIndex)
{
TTool *tool = TTool::getApplication()->getCurrentTool()->getTool();
if (!tool)
return;
TVectorImageP img(tool->getImage(true));
if (!img)
return;
TPalette *palette = img->getPalette();
TColorStyle *cs = palette->getStyle(styleIndex);
if (!cs->isStrokeStyle())
return;
if (m_indexes.empty())
return;
UndoSetStrokeStyle *undo = new UndoSetStrokeStyle(img, styleIndex);
std::set<int>::iterator it;
for (it = m_indexes.begin(); it != m_indexes.end(); ++it) {
int index = *it;
assert(0 <= index && index < (int)img->getStrokeCount());
TStroke *stroke = img->getStroke(index);
undo->addStroke(stroke);
stroke->setStyle(styleIndex);
}
tool->notifyImageChanged();
TUndoManager::manager()->add(undo);
}