Blob Blame Raw


// TnzCore includes
#include "tconst.h"
#include "tundo.h"

// TnzBase includes
#include "tfxattributes.h"

// TnzLib includes
#include "toonz/tcolumnfx.h"
#include "toonz/fxcommand.h"
#include "toonz/fxdag.h"
#include "toonz/txsheet.h"
#include "toonz/tfxhandle.h"
#include "toonz/tcolumnfxset.h"
#include "toonz/txsheethandle.h"
#include "toonzqt/fxschematicscene.h"

// TnzQt includes
#include "toonzqt/schematicnode.h"
#include "toonzqt/fxschematicnode.h"
#include "toonzqt/selectioncommandids.h"
#include "fxdata.h"

// Qt includes
#include <QApplication>
#include <QClipboard>

#include "toonzqt/fxselection.h"

namespace
{
bool canGroup(TFx *fx)
{
	TXsheetFx *xfx = dynamic_cast<TXsheetFx *>(fx);
	TOutputFx *ofx = dynamic_cast<TOutputFx *>(fx);
	return (!xfx && !ofx);
}
}

//=========================================================
//
// FxSelection
//
//=========================================================

FxSelection::FxSelection()
	: m_xshHandle(0), m_fxHandle(0), m_pastePosition(TConst::nowhere), m_schematicScene(0)
{
}

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

FxSelection::FxSelection(const FxSelection &src)
	: m_selectedFxs(src.m_selectedFxs), m_selectedLinks(src.m_selectedLinks), m_xshHandle(src.m_xshHandle), m_fxHandle(src.m_fxHandle), m_selectedColIndexes(src.m_selectedColIndexes), m_pastePosition(TConst::nowhere), m_schematicScene(src.m_schematicScene)
{
}

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

FxSelection::~FxSelection()
{
}

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

void FxSelection::enableCommands()
{
	enableCommand(this, MI_Clear, &FxSelection::deleteSelection);
	enableCommand(this, MI_Cut, &FxSelection::cutSelection);
	enableCommand(this, MI_Copy, &FxSelection::copySelection);
	enableCommand(this, MI_Paste, &FxSelection::pasteSelection);
	enableCommand(this, MI_Group, &FxSelection::groupSelection);
	enableCommand(this, MI_Ungroup, &FxSelection::ungroupSelection);
	enableCommand(this, MI_Collapse, &FxSelection::collapseSelection);
	enableCommand(this, MI_ExplodeChild, &FxSelection::explodeChild);
}

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

TSelection *FxSelection::clone() const
{
	assert(0);
	return new FxSelection(*this);
}

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

void FxSelection::select(TFxP fx)
{
	m_selectedFxs.append(fx);
}

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

void FxSelection::select(int colIndex)
{
	m_selectedColIndexes.append(colIndex);
	qSort(m_selectedColIndexes);
}

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

void FxSelection::unselect(int colIndex)
{
	m_selectedColIndexes.removeOne(colIndex);
}

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

void FxSelection::unselect(TFxP fx)
{
	int index = m_selectedFxs.indexOf(fx);
	if (index >= 0)
		m_selectedFxs.removeAt(index);
}

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

void FxSelection::select(SchematicLink *link)
{
	if (link->isLineShaped())
		return;
	Link boundingFxs = getBoundingFxs(link);
	if (boundingFxs == Link())
		return;
	m_selectedLinks.append(boundingFxs);
}

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

void FxSelection::unselect(SchematicLink *link)
{
	Link boundingFxs = getBoundingFxs(link);
	int index = m_selectedLinks.indexOf(boundingFxs);
	if (index >= 0)
		m_selectedLinks.removeAt(index);
}

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

bool FxSelection::isSelected(TFxP fx) const
{
	int i;
	for (i = 0; i < m_selectedFxs.size(); i++) {
		TFx *selectedFx = m_selectedFxs[i].getPointer();
		TZeraryColumnFx *zfx = dynamic_cast<TZeraryColumnFx *>(selectedFx);
		if (zfx && (fx.getPointer() == zfx || fx.getPointer() == zfx->getZeraryFx()))
			return true;
		if (fx.getPointer() == selectedFx)
			return true;
	}
	return false;
}

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

bool FxSelection::isSelected(int columnIndex) const
{
	return m_selectedColIndexes.contains(columnIndex);
}

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

bool FxSelection::isSelected(SchematicLink *link)
{
	Link boundingFxs = getBoundingFxs(link);
	return m_selectedLinks.contains(boundingFxs);
}

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

void FxSelection::deleteSelection()
{
	std::list<TFxP, std::allocator<TFxP>> fxList = m_selectedFxs.toStdList();
	TFxCommand::deleteSelection(fxList, m_selectedLinks.toStdList(), m_selectedColIndexes.toStdList(),
								m_xshHandle, m_fxHandle);
}

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

void FxSelection::copySelection()
{
	QClipboard *clipboard = QApplication::clipboard();
	FxsData *fxsData = new FxsData();
	fxsData->setFxs(m_selectedFxs, m_selectedLinks, m_selectedColIndexes, m_xshHandle->getXsheet());
	clipboard->setMimeData(fxsData);
}

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

void FxSelection::cutSelection()
{
	copySelection();
	deleteSelection();
}

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

void FxSelection::pasteSelection()
{
	/*--- Fxノードを1つだけ選択していた場合、Replace paste ---*/
	if (m_selectedFxs.size() >= 1 && m_selectedLinks.size() == 0 && m_selectedColIndexes.isEmpty())
		replacePasteSelection();
	/*--- Linkを1つだけ選択していた場合、Insert paste ---*/
	else if (m_selectedFxs.size() == 0 && m_selectedLinks.size() >= 1 && m_selectedColIndexes.isEmpty())
		insertPasteSelection();
	else {
		QClipboard *clipboard = QApplication::clipboard();
		const FxsData *fxsData = dynamic_cast<const FxsData *>(clipboard->mimeData());
		if (!fxsData)
			return;
		QList<TFxP> fxs;
		QMap<TFx *, int> zeraryFxColumnSize;
		QList<TXshColumnP> columns;
		fxsData->getFxs(fxs, zeraryFxColumnSize, columns);
		if (fxs.isEmpty() && columns.isEmpty())
			return;

		// in case of the paste command triggered from short cut key
		if (m_pastePosition == TConst::nowhere && m_schematicScene) {
			SchematicSceneViewer *ssv = dynamic_cast<SchematicSceneViewer *>(m_schematicScene->views().at(0));
			if (ssv)
				m_pastePosition = TPointD(ssv->getOldScenePos().x(), ssv->getOldScenePos().y());
		}

		TFxCommand::pasteFxs(fxs.toStdList(), zeraryFxColumnSize.toStdMap(), columns.toStdList(), m_pastePosition, m_xshHandle, m_fxHandle);

		if (m_schematicScene) {
			selectNone();
			for (int i = 0; i < (int)fxs.size(); i++)
				select(fxs[i]);
			m_schematicScene->selectNodes(fxs);
		}
	}
	m_pastePosition = TConst::nowhere;
}

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

bool FxSelection::insertPasteSelection()
{
	QClipboard *clipboard = QApplication::clipboard();
	const FxsData *fxsData = dynamic_cast<const FxsData *>(clipboard->mimeData());

	m_pastePosition = TConst::nowhere;

	if (!fxsData || !fxsData->isConnected())
		return false;

	if (m_selectedLinks.isEmpty())
		return true;

	// Start an undo block and ensure it is appropriately destroyed
	struct Auto {
		bool m_destruct;
		~Auto()
		{
			if (m_destruct)
				TUndoManager::manager()->endBlock();
		}
	} auto_ = {false};

	// Need to make a temporary copy of selected links. It's necessary since the selection will
	// be updated (cleared) after each insertion.
	QList<TFxCommand::Link> selectedLinks(m_selectedLinks);

	int i, size = selectedLinks.size();
	for (i = 0; i < size; ++i) {
		// Clone the fxs to be inserted
		QList<TFxP> fxs;
		QMap<TFx *, int> zeraryFxColumnSize;
		QList<TXshColumnP> columns;

		fxsData->getFxs(fxs, zeraryFxColumnSize, columns);
		if (fxs.empty() && columns.empty())
			return true;

		if (!auto_.m_destruct)
			auto_.m_destruct = true, TUndoManager::manager()->beginBlock();

		TFxCommand::insertPasteFxs(selectedLinks[i], fxs.toStdList(), zeraryFxColumnSize.toStdMap(),
								   columns.toStdList(), m_xshHandle, m_fxHandle);
	}

	return true;
}

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

bool FxSelection::addPasteSelection()
{
	QClipboard *clipboard = QApplication::clipboard();
	const FxsData *fxsData = dynamic_cast<const FxsData *>(clipboard->mimeData());

	m_pastePosition = TConst::nowhere;

	if (!fxsData || !fxsData->isConnected())
		return false;

	if (m_selectedFxs.isEmpty())
		return true;

	struct Auto {
		bool m_destruct;
		~Auto()
		{
			if (m_destruct)
				TUndoManager::manager()->endBlock();
		}
	} auto_ = {false};

	QList<TFxP> selectedFxs(m_selectedFxs);

	int i, size = selectedFxs.size();
	for (i = 0; i < size; ++i) {
		// Clone the fxs to be inserted
		QList<TFxP> fxs;
		QMap<TFx *, int> zeraryFxColumnSize;
		QList<TXshColumnP> columns;

		fxsData->getFxs(fxs, zeraryFxColumnSize, columns);
		if (fxs.empty() && columns.empty())
			return true;

		if (!auto_.m_destruct)
			auto_.m_destruct = true, TUndoManager::manager()->beginBlock();

		TFx *inFx = selectedFxs[i].getPointer();
		TFxCommand::addPasteFxs(inFx, fxs.toStdList(), zeraryFxColumnSize.toStdMap(),
								columns.toStdList(), m_xshHandle, m_fxHandle);
	}

	return true;
}

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

bool FxSelection::replacePasteSelection()
{
	QClipboard *clipboard = QApplication::clipboard();
	const FxsData *fxsData = dynamic_cast<const FxsData *>(clipboard->mimeData());

	m_pastePosition = TConst::nowhere;

	if (!fxsData || !fxsData->isConnected())
		return false;

	if (m_selectedFxs.isEmpty())
		return true;

	struct Auto {
		bool m_destruct;
		~Auto()
		{
			if (m_destruct)
				TUndoManager::manager()->endBlock();
		}
	} auto_ = {false};

	QList<TFxP> selectedFxs(m_selectedFxs);

	int i, size = selectedFxs.size();
	for (i = 0; i < size; ++i) {
		// Clone the fxs to be inserted
		QList<TFxP> fxs;
		QMap<TFx *, int> zeraryFxColumnSize;
		QList<TXshColumnP> columns;

		fxsData->getFxs(fxs, zeraryFxColumnSize, columns);
		if (fxs.empty() && columns.empty())
			return true;

		if (!auto_.m_destruct)
			auto_.m_destruct = true, TUndoManager::manager()->beginBlock();

		TFx *inFx = m_selectedFxs[i].getPointer();
		TFxCommand::replacePasteFxs(inFx, fxs.toStdList(), zeraryFxColumnSize.toStdMap(),
									columns.toStdList(), m_xshHandle, m_fxHandle);
	}

	return true;
}

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

void FxSelection::groupSelection()
{
	if (m_selectedFxs.size() <= 1)
		return;
	TFxCommand::groupFxs(m_selectedFxs.toStdList(), m_xshHandle);
	selectNone();
	m_xshHandle->notifyXsheetChanged();
}

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

void FxSelection::ungroupSelection()
{
	if (isEmpty())
		return;
	QSet<int> idSet;
	int i;
	for (i = 0; i < m_selectedFxs.size(); i++) {
		int groupId = m_selectedFxs[i]->getAttributes()->getGroupId();
		if (groupId > 0)
			idSet.insert(groupId);
	}

	TUndoManager::manager()->beginBlock();
	QSet<int>::iterator it;
	for (it = idSet.begin(); it != idSet.end(); it++)
		TFxCommand::ungroupFxs(*it, m_xshHandle);
	TUndoManager::manager()->endBlock();
	selectNone();
	m_xshHandle->notifyXsheetChanged();
}

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

void FxSelection::collapseSelection()
{
	if (!m_selectedFxs.isEmpty())
		emit doCollapse(m_selectedFxs);
}

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

void FxSelection::explodeChild()
{
	if (!m_selectedFxs.isEmpty())
		emit doExplodeChild(m_selectedFxs);
}

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

Link FxSelection::getBoundingFxs(SchematicLink *link)
{
	Link boundingFxs;
	if (link) {
		SchematicPort *port = link->getStartPort();
		if (!port)
			return boundingFxs;
		if (port->getType() == 201 || port->getType() == 202 || port->getType() == 203)
			boundingFxs = getBoundingFxs(port, link->getOtherPort(port));
		else if (port->getType() == 200 || port->getType() == 204)
			boundingFxs = getBoundingFxs(link->getOtherPort(port), port);
	}
	return boundingFxs;
}

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

Link FxSelection::getBoundingFxs(SchematicPort *inputPort, SchematicPort *outputPort)
{
	Link boundingFxs;
	FxSchematicNode *inputNode = dynamic_cast<FxSchematicNode *>(outputPort->getNode());
	FxSchematicNode *outputNode = dynamic_cast<FxSchematicNode *>(inputPort->getNode());
	FxGroupNode *groupNode = dynamic_cast<FxGroupNode *>(inputNode);

	if (!inputNode || !outputNode || (groupNode && groupNode->getOutputConnectionsCount() != 1))
		return boundingFxs;
	if (dynamic_cast<TXsheetFx *>(outputNode->getFx())) {
		if (!groupNode)
			boundingFxs.m_inputFx = inputNode->getFx();
		else {
			TFxSet *terminals = m_xshHandle->getXsheet()->getFxDag()->getTerminalFxs();
			QList<TFxP> roots = groupNode->getRootFxs();
			int i;
			for (i = 0; i < roots.size(); i++)
				if (terminals->containsFx(roots[i].getPointer())) {
					boundingFxs.m_inputFx = roots[i];
					break;
				}
		}
		boundingFxs.m_outputFx = outputNode->getFx();
		return boundingFxs;
	}

	if (outputNode->isA(eGroupedFx)) {
		//devo prima trovare l'effetto interno al gruppo al quale inputNode e' linkato.
		FxGroupNode *groupNode = dynamic_cast<FxGroupNode *>(outputNode);
		assert(groupNode);
		QList<TFx *> fxs;
		TFx *inputFx = inputNode->getFx();
		int i;
		for (i = 0; i < inputFx->getOutputConnectionCount(); i++) {
			TFx *outputFx = inputFx->getOutputConnection(i)->getOwnerFx();
			if (!outputFx)
				continue;
			if (groupNode->contains(outputFx))
				fxs.push_back(outputFx);
		}
		if (fxs.size() != 1) //un nodo esterno al gruppo puo' essere linkato a piu' nodi interni al gruppo
			return boundingFxs;

		TFx *outputFx = fxs[0];
		//ho tovato l'effetto, ora devo trovare l'indice della porta a cui e' linkato l'effetto in input
		for (i = 0; i < outputFx->getInputPortCount(); i++) {
			TFxPort *inputPort = outputFx->getInputPort(i);
			TFx *fx = inputPort->getFx();
			if (fx == inputFx)
				break;
		}
		if (i >= outputFx->getInputPortCount())
			return boundingFxs;
		boundingFxs.m_inputFx = inputFx;
		boundingFxs.m_outputFx = outputFx;
		boundingFxs.m_index = i;
		return boundingFxs;
	} else {
		bool found = false;
		int i, index = -1;
		for (i = 0; i < outputNode->getInputPortCount() && !found; i++) {
			FxSchematicPort *inputAppPort = outputNode->getInputPort(i);
			int j;
			for (j = 0; j < inputAppPort->getLinkCount(); j++) {
				FxSchematicNode *outputAppNode = dynamic_cast<FxSchematicNode *>(inputAppPort->getLinkedNode(j));
				if (!outputAppNode)
					continue;
				FxSchematicPort *outputAppPort = outputAppNode->getOutputPort();
				if (inputAppPort == inputPort && outputPort == outputAppPort) {
					found = true;
					index = i;
					break;
				}
			}
		}
		if (index == -1)
			return boundingFxs;
		TFx *inputFx = inputNode->getFx();
		TFx *outputFx = outputNode->getFx();
		boundingFxs.m_inputFx = inputFx;
		boundingFxs.m_outputFx = outputFx;
		boundingFxs.m_index = index;
		return boundingFxs;
	}
}

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

bool FxSelection::isConnected()
{
	if (m_selectedFxs.isEmpty())
		return false;
	QList<TFx *> visitedFxs;
	visitFx(m_selectedFxs.at(0).getPointer(), visitedFxs);
	bool connected = true;
	QList<TFxP>::const_iterator it;
	TXsheet *xsh = m_xshHandle->getXsheet();
	TFxSet *internalFxs = xsh->getFxDag()->getInternalFxs();
	for (it = m_selectedFxs.begin(); it != m_selectedFxs.end(); it++) {
		TFx *selectedFx = it->getPointer();
		TColumnFx *cfx = dynamic_cast<TColumnFx *>(selectedFx);
		if (!cfx && !internalFxs->containsFx(selectedFx))
			return false;
		TZeraryColumnFx *zfx = dynamic_cast<TZeraryColumnFx *>(selectedFx);
		if (zfx)
			selectedFx = zfx->getZeraryFx();
		connected = connected && visitedFxs.contains(selectedFx);
	}
	return connected;
}

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

void FxSelection::visitFx(TFx *fx, QList<TFx *> &visitedFxs)
{
	if (visitedFxs.contains(fx))
		return;
	TZeraryColumnFx *zfx = dynamic_cast<TZeraryColumnFx *>(fx);
	if (zfx)
		fx = zfx->getZeraryFx();
	if (!canGroup(fx))
		return;
	visitedFxs.append(fx);
	int i;
	for (i = 0; i < fx->getInputPortCount(); i++) {
		TFx *inputFx = fx->getInputPort(i)->getFx();
		TZeraryColumnFx *onputZFx = dynamic_cast<TZeraryColumnFx *>(inputFx);
		if (onputZFx)
			inputFx = onputZFx->getZeraryFx();
		if (!inputFx)
			continue;
		bool canBeGrouped = !inputFx->getAttributes()->isGrouped() ||
							(inputFx->getAttributes()->getEditingGroupId() == fx->getAttributes()->getEditingGroupId());
		if (!visitedFxs.contains(inputFx) && isSelected(inputFx) && canBeGrouped)
			visitFx(inputFx, visitedFxs);
	}
	if (zfx)
		fx = zfx;
	if (fx->isZerary() && !zfx) {
		TXsheet *xsh = m_xshHandle->getXsheet();
		int columnCount = xsh->getColumnCount();
		int j;
		for (j = 0; j < columnCount; j++) {
			TZeraryColumnFx *zerary = dynamic_cast<TZeraryColumnFx *>(xsh->getColumn(j)->getFx());
			if (zerary && zerary->getZeraryFx() == fx) {
				fx = zerary;
				break;
			}
		}
	}
	for (i = 0; i < fx->getOutputConnectionCount(); i++) {
		TFx *outputFx = fx->getOutputConnection(i)->getOwnerFx();
		if (!outputFx)
			continue;
		bool canBeGrouped = !outputFx->getAttributes()->isGrouped() ||
							(outputFx->getAttributes()->getEditingGroupId() == fx->getAttributes()->getEditingGroupId());
		if (!visitedFxs.contains(outputFx) && isSelected(outputFx) && canBeGrouped)
			visitFx(outputFx, visitedFxs);
	}
}

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

bool FxSelection::areLinked(TFx *outFx, TFx *inFx)
{
	int i;
	for (i = 0; i < outFx->getInputPortCount(); i++) {
		TFx *inputFx = outFx->getInputPort(i)->getFx();
		if (inFx == inputFx)
			return true;
	}
	return false;
}