#include "toonz/fxdag.h"
// TnzLib includes
#include "toonz/tcolumnfxset.h"
#include "toonz/tcolumnfx.h"
#include "tw/stringtable.h"
// TnzBase includes
#include "tmacrofx.h"
#include "tfxattributes.h"
// TnzCore includes
#include "tstream.h"
// tcg includes
#include "tcg/function_types.h"
// STD includes
//#include <cwctypes>
//===================================================================
FxDag::FxDag()
: m_internalFxs(new TFxSet()), m_terminalFxs(new TFxSet()), m_groupIdCount(0), m_dagGridDimension(eSmall)
{
TXsheetFx *xsheetFx = new TXsheetFx;
xsheetFx->setFxDag(this);
m_xsheetFx = xsheetFx;
m_xsheetFx->addRef();
m_xsheetFx->setNewIdentifier();
addOutputFx();
m_outputFxs[0]->getInputPort(0)->setFx(m_xsheetFx);
}
//-------------------------------------------------------------------
FxDag::~FxDag()
{
delete m_internalFxs;
delete m_terminalFxs;
m_xsheetFx->release();
int i;
for (i = 0; i < (int)m_outputFxs.size(); i++)
m_outputFxs[i]->release();
}
//-------------------------------------------------------------------
TOutputFx *FxDag::addOutputFx(TOutputFx *outputFx)
{
if (!outputFx)
outputFx = new TOutputFx();
outputFx->addRef();
m_xsheetFx->setNewIdentifier();
assert(outputFx->getInputPortCount() == 1);
m_outputFxs.push_back(outputFx);
return outputFx;
}
//-------------------------------------------------------------------
void FxDag::removeOutputFx(TOutputFx *fx)
{
assert(fx);
if (m_outputFxs.size() == 1)
return;
if (std::find(m_outputFxs.begin(), m_outputFxs.end(), fx) == m_outputFxs.end())
return;
m_outputFxs.erase(
std::remove(m_outputFxs.begin(), m_outputFxs.end(), fx),
m_outputFxs.end());
fx->release();
}
//-------------------------------------------------------------------
TFxSet *FxDag::getInternalFxs() const
{
return m_internalFxs;
}
//-------------------------------------------------------------------
void FxDag::setCurrentOutputFx(TOutputFx *fx)
{
std::vector<TOutputFx *>::iterator it;
it = std::find(m_outputFxs.begin(), m_outputFxs.end(), fx);
if (it == m_outputFxs.end())
return;
if (it == m_outputFxs.begin())
return;
std::swap(*it, *m_outputFxs.begin());
}
TOutputFx *FxDag::getCurrentOutputFx() const
{
assert(!m_outputFxs.empty());
return m_outputFxs[0];
}
//-------------------------------------------------------------------
bool FxDag::checkLoop(TFx *inputFx, TFx *fx)
{
if (inputFx == fx)
return true;
if (dynamic_cast<TXsheetFx *>(inputFx)) {
TFxSet *terminals = getTerminalFxs();
for (int i = 0; i < terminals->getFxCount(); i++) {
TFx *tfx = terminals->getFx(i);
if (tfx && checkLoop(tfx, fx))
return true;
}
} else {
if (TZeraryColumnFx *zerary = dynamic_cast<TZeraryColumnFx *>(inputFx))
inputFx = zerary->getZeraryFx();
for (int i = 0; i < inputFx->getInputPortCount(); i++) {
TFx *ifx = inputFx->getInputPort(i)->getFx();
if (ifx && checkLoop(ifx, fx))
return true;
}
}
return false;
}
//-------------------------------------------------------------------
TFxSet *FxDag::getTerminalFxs() const
{
return m_terminalFxs;
}
//-------------------------------------------------------------------
void FxDag::addToXsheet(TFx *fx)
{
m_terminalFxs->addFx(fx);
}
//-------------------------------------------------------------------
void FxDag::removeFromXsheet(TFx *fx)
{
m_terminalFxs->removeFx(fx);
}
//-------------------------------------------------------------------
namespace
{
struct NotAlnum {
bool operator()(wint_t val) const { return !iswalnum(val); }
};
} // namespace
void FxDag::assignUniqueId(TFx *fx)
{
struct locals {
static void eraseNonAlnums(std::wstring &str)
{
str.erase(std::remove_if(str.begin(), str.end(), NotAlnum()), str.end());
}
}; // locals
string type = fx->getFxType();
int count = ++m_typeTable[type];
fx->getAttributes()->setId(count);
wstring name = TStringTable::translate(type);
locals::eraseNonAlnums(name); // fx ids are used as XML tag names - thus, we'll restrict
// the char set to alnums. Specifically, '/' must be ruled out.
// E.g.: "Erode/Dilate 1" must become "ErodeDilate1"
name = name + QString::number(count).rightJustified(2, '0').toStdWString();
if (fx->getName() == L"")
fx->setName(name);
fx->setFxId(name);
m_idTable[toLower(name)] = fx;
}
//-------------------------------------------------------------------
TFx *FxDag::getFxById(wstring id) const
{
std::map<wstring, TFx *>::const_iterator it = m_idTable.find(id);
if (it == m_idTable.end())
return 0;
else
return it->second;
}
//-------------------------------------------------------------------
void FxDag::updateFxTypeTable(TFx *fx, int value)
{
string type = fx->getFxType();
m_typeTable[type] = value;
}
//-------------------------------------------------------------------
void FxDag::updateFxIdTable(TFx *fx)
{
m_idTable[toLower(fx->getFxId())] = fx;
}
//-------------------------------------------------------------------
int FxDag::getFxTypeCount(TFx *fx)
{
string type = fx->getFxType();
std::map<string, int>::iterator it = m_typeTable.find(type);
if (it == m_typeTable.end())
return 0;
return it->second;
}
//-------------------------------------------------------------------
void FxDag::getFxs(vector<TFx *> &fxs) const
{
std::set<TFx *> fxSet;
getInternalFxs()->getFxs(fxSet);
fxs.insert(fxs.end(), fxSet.begin(), fxSet.end());
}
//-------------------------------------------------------------------
bool FxDag::isRendered(TFx *fx) const
{
if (m_terminalFxs->containsFx(fx))
return true;
if (dynamic_cast<TOutputFx *>(fx))
return true;
int i;
for (i = 0; i < fx->getOutputConnectionCount(); i++) {
TFx *outFx = fx->getOutputConnection(i)->getOwnerFx();
if (outFx && isRendered(outFx))
return true;
}
return false;
}
//-------------------------------------------------------------------
bool FxDag::isControl(TFx *fx) const
{
if (m_terminalFxs->containsFx(fx))
return false;
if (dynamic_cast<TOutputFx *>(fx))
return false;
int i;
for (i = 0; i < fx->getOutputConnectionCount(); i++) {
TFxPort *port = fx->getOutputConnection(i);
TFx *outFx = port->getOwnerFx();
if (outFx) {
if (outFx->getInputPort(0) != port)
return true;
if (isControl(outFx))
return true;
}
}
return false;
}
//-------------------------------------------------------------------
namespace
{
TFx *search(const std::map<TFx *, TFx *> &table, TFx *fx)
{
std::map<TFx *, TFx *>::const_iterator it = table.find(fx);
return it == table.end() ? 0 : it->second;
}
}
//-------------------------------------------------------------------
void FxDag::saveData(TOStream &os, int occupiedColumnCount)
{
if (getInternalFxs()->getFxCount() > 0) {
os.openChild("internal");
getInternalFxs()->saveData(os, occupiedColumnCount);
os.closeChild();
}
if (getTerminalFxs()->getFxCount() > 0) {
os.openChild("terminal");
getTerminalFxs()->saveData(os, occupiedColumnCount);
os.closeChild();
}
os.child("xsheet") << m_xsheetFx;
int k;
for (k = 0; k < (int)m_outputFxs.size(); k++)
os.child("output") << m_outputFxs[k];
os.child("grid_dimension") << (int)m_dagGridDimension;
}
//-------------------------------------------------------------------
void FxDag::loadData(TIStream &is)
{
VersionNumber tnzVersion = is.getVersion();
int k;
for (k = 0; k < (int)m_outputFxs.size(); k++)
m_outputFxs[k]->release();
m_outputFxs.clear();
string tagName;
while (is.openChild(tagName)) {
if (tagName == "terminal") {
TFxSet *fxSet = getTerminalFxs();
fxSet->loadData(is);
int i;
for (i = 0; i < fxSet->getFxCount(); i++) {
TFx *fx = fxSet->getFx(i);
if (fx->getAttributes()->isGrouped() && m_groupIdCount < fx->getAttributes()->getGroupId())
m_groupIdCount = fx->getAttributes()->getGroupId();
if (TZeraryColumnFx *zfx = dynamic_cast<TZeraryColumnFx *>(fx))
fx = zfx->getZeraryFx();
if (tnzVersion < VersionNumber(1, 16)) {
wstring app = fx->getName();
assignUniqueId(fx);
fx->setName(app);
continue;
}
int fxTypeCount = getFxTypeCount(fx);
int maxFxTypeId = tmax(fxTypeCount, fx->getAttributes()->getId());
updateFxTypeTable(fx, maxFxTypeId);
TMacroFx *macroFx = dynamic_cast<TMacroFx *>(fx);
if (macroFx) {
vector<TFxP> fxs = macroFx->getFxs();
int j;
for (j = 0; j < (int)fxs.size(); j++) {
TFxP inMacroFx = fxs[j];
fxTypeCount = getFxTypeCount(inMacroFx.getPointer());
maxFxTypeId = tmax(fxTypeCount, inMacroFx->getAttributes()->getId());
updateFxTypeTable(inMacroFx.getPointer(), maxFxTypeId);
m_idTable[toLower(inMacroFx->getFxId())] = inMacroFx.getPointer();
}
}
}
} else if (tagName == "internal") {
TFxSet *fxSet = getInternalFxs();
fxSet->loadData(is);
int i;
for (i = 0; i < fxSet->getFxCount(); i++) {
TFx *fx = fxSet->getFx(i);
if (fx->getAttributes()->isGrouped() && m_groupIdCount < fx->getAttributes()->getGroupId())
m_groupIdCount = fx->getAttributes()->getGroupId();
if (TZeraryColumnFx *zfx = dynamic_cast<TZeraryColumnFx *>(fx))
fx = zfx->getZeraryFx();
if (tnzVersion < VersionNumber(1, 16)) {
wstring app = fx->getName();
assignUniqueId(fx);
fx->setName(app);
continue;
}
int fxTypeCount = getFxTypeCount(fx);
int maxFxTypeId = tmax(fxTypeCount, fx->getAttributes()->getId());
updateFxTypeTable(fx, maxFxTypeId);
m_idTable[toLower(fx->getFxId())] = fx;
}
} else if (tagName == "xsheet" || tagName == "output") {
TPersist *p = 0;
is >> p;
TFx *fx = dynamic_cast<TFx *>(p);
if (!fx)
throw TException("FxDag. unexpeced Fx");
fx->addRef();
fx->setNewIdentifier();
if (tagName == "xsheet") {
m_xsheetFx->release();
m_xsheetFx = fx;
TXsheetFx *xsheetFx = dynamic_cast<TXsheetFx *>(fx);
if (xsheetFx)
xsheetFx->setFxDag(this);
} else {
TOutputFx *outputFx = dynamic_cast<TOutputFx *>(fx);
if (outputFx)
m_outputFxs.push_back(outputFx);
}
} else if (tagName == "grid_dimension") {
is >> m_dagGridDimension;
} else
throw TException("FxDag. unexpeced tag: " + tagName);
is.closeChild();
}
}