Blob Blame Raw


#include "hookselection.h"

#include "toonz/txshlevelhandle.h"
#include "toonz/txshsimplelevel.h"
#include "toonz/txsheethandle.h"
#include "toonz/tstageobjecttree.h"

#include "tools/tool.h"
#include "tools/toolhandle.h"

#include "toonzqt/selectioncommandids.h"

#include <QApplication>
#include <QClipboard>

//=============================================================================
//
// HookUndo
//
//=============================================================================

HookUndo::HookUndo(const TXshLevelP &level)
	: m_level(level)
{
	HookSet *hookSet = m_level->getHookSet();
	assert(hookSet);
	if (hookSet)
		m_oldHooks = *hookSet;
}

//---------------------------------------------------------------------------

HookUndo::~HookUndo()
{
}

//---------------------------------------------------------------------------

void HookUndo::onAdd()
{
	HookSet *hookSet = m_level->getHookSet();
	assert(hookSet);
	if (hookSet)
		m_newHooks = *hookSet;
}

//---------------------------------------------------------------------------

void HookUndo::assignHookSet(const HookSet &src) const
{
	HookSet *hookSet = m_level->getHookSet();
	assert(hookSet);
	if (hookSet)
		*hookSet = src;

	TTool::getApplication()->getCurrentXsheet()->getXsheet()->getStageObjectTree()->invalidateAll();
	TTool *tool = TTool::getApplication()->getCurrentTool()->getTool();
	if (tool) {
		tool->updateMatrix();
		tool->invalidate();
	}
}

//---------------------------------------------------------------------------

void HookUndo::undo() const
{
	assignHookSet(m_oldHooks);
}

//---------------------------------------------------------------------------

void HookUndo::redo() const
{
	assignHookSet(m_newHooks);
}

//---------------------------------------------------------------------------

int HookUndo::getSize() const
{
	return sizeof(*this) + 2 * sizeof(HookSet);
}

//=============================================================================
//
// HooksData
//
//=============================================================================

HooksData::HooksData(const TXshLevelP &level)
	: m_level(level)
{
}

//---------------------------------------------------------------------------

HooksData::~HooksData()
{
}

//---------------------------------------------------------------------------

HooksData *HooksData::clone() const
{
	HooksData *newData = new HooksData(m_level);
	newData->m_hookPositions = m_hookPositions;
	return newData;
}

//---------------------------------------------------------------------------

void HooksData::storeHookPositions(const std::vector<int> &ids)
{
	if (ids.empty())
		return;
	TTool::Application *app = TTool::getApplication();
	TXshLevelP level = app->getCurrentLevel()->getLevel();
	assert(level = m_level);
	if (level != m_level || !m_level || m_level->getSimpleLevel()->isReadOnly())
		return;
	HookSet *hookSet = m_level->getHookSet();
	if (!hookSet)
		return;

	TFrameId fid = app->getCurrentTool()->getTool()->getCurrentFid();
	int i, idsSize = ids.size();
	m_hookPositions.clear();
	for (i = 0; i < idsSize; i++) {
		Hook *hook = hookSet->getHook(ids[i]);
		assert(hook);
		if (!hook)
			continue;
		TPointD aPos = hook->getAPos(fid);
		TPointD bPos = hook->getBPos(fid);
		m_hookPositions.push_back(HookPosition(ids[i], aPos, bPos));
	}
}

//---------------------------------------------------------------------------

void HooksData::restoreHookPositions() const
{
	if (m_hookPositions.empty())
		return;
	TTool::Application *app = TTool::getApplication();
	TXshLevelP level = app->getCurrentLevel()->getLevel();
	assert(level = m_level);
	if (level != m_level || !m_level || m_level->getSimpleLevel()->isReadOnly())
		return;
	HookSet *hookSet = m_level->getHookSet();
	if (!hookSet)
		return;

	TFrameId fid = app->getCurrentTool()->getTool()->getCurrentFid();
	int i, posSize = m_hookPositions.size();

	for (i = 0; i < posSize; i++) {
		HookPosition hookPos = m_hookPositions[i];
		Hook *hook = hookSet->getHook(hookPos.m_id);
		assert(hook);
		if (!hook)
			continue;
		hook->setAPos(fid, hookPos.m_aPos);
		hook->setBPos(fid, hookPos.m_bPos);
	}
}

//===========================================================================
//
// HookSelection
//
//===========================================================================

HookSelection::HookSelection()
{
}

//---------------------------------------------------------------------------

HookSelection::~HookSelection()
{
}

//---------------------------------------------------------------------------

TSelection *HookSelection::clone() const
{
	return new HookSelection(*this);
}

//---------------------------------------------------------------------------

void HookSelection::select(int id, int side)
{
	m_hooks.insert(std::make_pair(id, side));
}

//---------------------------------------------------------------------------

void HookSelection::unselect(int id, int side)
{
	m_hooks.erase(std::make_pair(id, side));
}

//---------------------------------------------------------------------------

bool HookSelection::isSelected(int id, int side) const
{
	return m_hooks.count(std::make_pair(id, side)) > 0;
}

//---------------------------------------------------------------------------

void HookSelection::invertSelection(int id, int side)
{
	if (isSelected(id, side))
		unselect(id, side);
	else
		select(id, side);
}

//---------------------------------------------------------------------------

bool HookSelection::isEmpty() const
{
	return m_hooks.empty();
}

//---------------------------------------------------------------------------

HookSet *HookSelection::getHookSet() const
{
	TXshLevel *xl = TTool::getApplication()->getCurrentLevel()->getLevel();
	if (!xl)
		return 0;
	return xl->getHookSet();
}

//---------------------------------------------------------------------------

bool HookSelection::select(const TSelection *s)
{
	if (const HookSelection *hs = dynamic_cast<const HookSelection *>(s)) {
		m_level = hs->m_level;
		m_hooks = hs->m_hooks;
		return true;
	} else
		return false;
}

//---------------------------------------------------------------------------

void HookSelection::enableCommands()
{
	enableCommand(this, MI_Clear, &HookSelection::deleteSelectedHooks);
	enableCommand(this, MI_Copy, &HookSelection::copySelectedHooks);
	enableCommand(this, MI_Cut, &HookSelection::cutSelectedHooks);
	enableCommand(this, MI_Paste, &HookSelection::pasteSelectedHooks);
}

//---------------------------------------------------------------------------

void HookSelection::deleteSelectedHooks()
{
	TTool::Application *app = TTool::getApplication();
	TTool *tool = app->getCurrentTool()->getTool();
	if (!app)
		return;
	TXshLevel *xl = app->getCurrentLevel()->getLevel();
	HookSet *hookSet = xl->getHookSet();
	if (!xl || !xl->getSimpleLevel() || !hookSet || xl->getSimpleLevel()->isReadOnly())
		return;

	HookUndo *undo = new HookUndo(xl->getSimpleLevel());
	TFrameId fid = tool->getCurrentFid();

	for (int i = 0; i < hookSet->getHookCount(); i++) {
		Hook *hook = hookSet->getHook(i);
		if (!hook || hook->isEmpty())
			continue;
		if (isSelected(i, 1) && isSelected(i, 2))
			hookSet->clearHook(hook);
		else if (isSelected(i, 2))
			hook->setBPos(fid, hook->getAPos(fid));
		else if (isSelected(i, 1))
			hook->setAPos(fid, hook->getBPos(fid));
	}
	TUndoManager::manager()->add(undo);
	app->getCurrentXsheet()->getXsheet()->getStageObjectTree()->invalidateAll();
	tool->invalidate();
}

//---------------------------------------------------------------------------

void HookSelection::copySelectedHooks()
{
	if (isEmpty())
		return;
	std::vector<int> ids;
	std::set<std::pair<int, int>>::iterator it;
	for (it = m_hooks.begin(); it != m_hooks.end(); it++) {
		if (std::find(ids.begin(), ids.end(), it->first) == ids.end())
			ids.push_back(it->first);
	}

	TXshLevel *xl = TTool::getApplication()->getCurrentLevel()->getLevel();
	HooksData *data = new HooksData(xl);
	data->storeHookPositions(ids);
	QApplication::clipboard()->setMimeData(data);
}

//---------------------------------------------------------------------------

void HookSelection::cutSelectedHooks()
{
	copySelectedHooks();
	TXshLevel *xl = TTool::getApplication()->getCurrentLevel()->getLevel();
	TUndo *undo = new HookUndo(xl);
	HookSet *hookSet = xl->getHookSet();
	std::set<std::pair<int, int>>::iterator it;
	for (it = m_hooks.begin(); it != m_hooks.end(); it++) {
		Hook *hook = hookSet->getHook(it->first);
		assert(hook);
		if (!hook)
			return;
		TFrameId fid = TTool::getApplication()->getCurrentTool()->getTool()->getCurrentFid();
		hook->eraseFrame(fid);
	}
	TUndoManager::manager()->add(undo);
	TTool::getApplication()->getCurrentTool()->getTool()->invalidate();
}

//---------------------------------------------------------------------------

void HookSelection::pasteSelectedHooks()
{
	const QMimeData *data = QApplication::clipboard()->mimeData();
	const HooksData *hooksData = dynamic_cast<const HooksData *>(data);
	if (!hooksData)
		return;
	TXshLevel *xl = TTool::getApplication()->getCurrentLevel()->getLevel();
	TUndo *undo = new HookUndo(xl);
	hooksData->restoreHookPositions();
	TUndoManager::manager()->add(undo);
	TTool::getApplication()->getCurrentTool()->getTool()->invalidate();
}