Blob Blame Raw


#include "toonz/childstack.h"
#include "toonz/toonzscene.h"
#include "toonz/txsheet.h"
#include "toonz/txshchildlevel.h"
#include "toonz/txshcell.h"
#include "toonz/txshleveltypes.h"
#include "toonz/scenefx.h"

//=============================================================================
//!The Node class is a container of element necessary to define a sub-xsheet.
/*!
   The class contain a pointer to \b TXsheet \b m_xsheet, two integer to identify column
   \b m_col and row \b m_row, a \b TXshChildLevelP \b m_cl and a bool \b m_justCreated.
*/

class ChildStack::Node
{
public:
	TXsheet *m_xsheet;
	int m_row, m_col;
	std::map<int, int> m_rowTable;
	TXshChildLevelP m_cl;
	bool m_justCreated;
	Node()
		: m_xsheet(0), m_row(0), m_col(0), m_rowTable(), m_justCreated(false)
	{
	}
};

//=============================================================================
// ChildStack

ChildStack::ChildStack(ToonzScene *scene)
	: m_scene(scene), m_xsheet(new TXsheet()), m_editInPlace(false)
{
	m_xsheet->setScene(m_scene);
	m_xsheet->addRef();
}

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

ChildStack::~ChildStack()
{
	m_xsheet->release();
	clearPointerContainer(m_stack);
}

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

void ChildStack::clear()
{
	m_xsheet->clearAll();
	m_xsheet->setScene(m_scene);

	clearPointerContainer(m_stack);
}

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

bool ChildStack::openChild(int row, int col)
{
	TXshChildLevel *childLevel = 0;
	TXshCell cell = m_xsheet->getCell(row, col);
	if (!cell.m_level) {
		return false;
	} else
		childLevel = cell.m_level->getChildLevel();
	if (!childLevel)
		return false;
	TXsheet *childXsheet = childLevel->getXsheet();
	Node *node = new Node();
	node->m_row = row;
	node->m_col = col;
	node->m_xsheet = m_xsheet;
	node->m_cl = childLevel;
	node->m_justCreated = !cell.m_level;
	int r0 = 0, r1 = -1;
	m_xsheet->getCellRange(col, r0, r1);
	for (int r = r0; r <= r1; r++) {
		TXshCell cell = m_xsheet->getCell(r, col);
		if (cell.m_level.getPointer() == childLevel) {
			int d = cell.m_frameId.getNumber() - 1;
			std::map<int, int>::iterator it = node->m_rowTable.find(d);
			if (it == node->m_rowTable.end())
				node->m_rowTable[d] = r;
		}
	}
	m_stack.push_back(node);
	m_xsheet = childXsheet;
	return true;
}

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

bool ChildStack::closeChild(int &row, int &col)
{
	if (m_stack.empty())
		return false;

	TXsheet *childXsh = m_xsheet;
	childXsh->updateFrameCount(); // non dovrebbe essere necessario, ma non si sa mai
	int childFrameCount = childXsh->getFrameCount();

	Node *node = m_stack.back();
	m_stack.pop_back();

	TXsheet *parentXsh = node->m_xsheet;
	TXshChildLevelP cl = node->m_cl;
	row = node->m_row;
	col = node->m_col;
	bool justCreated = node->m_justCreated;
	delete node;

	m_xsheet = parentXsh;
	m_xsheet->updateFrameCount();

	//if(cl) cl->invalidateIcon();

	TXsheet *xsh = m_xsheet;
	if (justCreated) {
		assert(xsh->getCell(row, col).m_level.getPointer() == cl.getPointer());
		if (childFrameCount > 1) {
			xsh->insertCells(row + 1, col, childFrameCount - 1);
			for (int i = 1; i < childFrameCount; i++)
				xsh->setCell(row + i, col, TXshCell(cl.getPointer(), TFrameId(1 + i)));
		}
	}

	return true;
}

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

TXshChildLevel *ChildStack::createChild(int row, int col)
{
	TXshLevel *xl = m_scene->createNewLevel(CHILD_XSHLEVEL);
	m_xsheet->setCell(row, col, TXshCell(xl, TFrameId(1)));

	TXshCell cell = m_xsheet->getCell(row, col);
	return cell.m_level->getChildLevel();
}

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

int ChildStack::getAncestorCount() const
{
	return m_stack.size();
}

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

TXsheet *ChildStack::getTopXsheet() const
{
	return m_stack.empty() ? m_xsheet : m_stack.front()->m_xsheet;
}

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

std::pair<TXsheet *, int> ChildStack::getAncestor(int row) const
{
	TXsheet *xsh = m_xsheet;
	int i = m_stack.size() - 1;
	while (i >= 0) {
		std::map<int, int>::const_iterator it = m_stack[i]->m_rowTable.find(row);
		if (it == m_stack[i]->m_rowTable.end())
			break;
		row = it->second;
		xsh = m_stack[i]->m_xsheet;
		i--;
	}
	return std::make_pair(xsh, row);
}

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

bool ChildStack::getAncestorAffine(TAffine &aff, int row) const
{
	aff = TAffine();
	int i = m_stack.size() - 1;
	while (i >= 0) {
		std::map<int, int>::const_iterator it = m_stack[i]->m_rowTable.find(row);
		if (it == m_stack[i]->m_rowTable.end())
			break;

		row = it->second;
		Node *node = m_stack[i];
		TAffine aff2;
		if (!getColumnPlacement(aff2, node->m_xsheet, row, node->m_col, false))
			return false;
		aff = aff2 * aff;
		i--;
	}
	return true;
}

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

void ChildStack::setEditInPlace(bool editInPlace)
{
	m_editInPlace = editInPlace;
}