Blob Blame Raw


#include "toonz/studiopalettecmd.h"
#include "toonz/tpalettehandle.h"
#include "toonz/txsheethandle.h"
#include "toonz/txshlevelhandle.h"
#include "tundo.h"
#include "tcolorstyles.h"
#include "tsystem.h"
#include "tconvert.h"
#include "ttoonzimage.h"
#include "timagecache.h"

#include "toonz/studiopalette.h"
#include "toonz/toonzscene.h"
#include "toonz/levelset.h"
#include "toonz/txshsimplelevel.h"
#include "toonz/txshleveltypes.h"
#include "toonz/sceneproperties.h"
#include "toonz/toonzfolders.h"
#include "toonz/txsheet.h"
#include <QApplication>

#include "historytypes.h"

/*! \namespace StudioPaletteCmd
		\brief Provides a collection of methods to manage \b StudioPalette.
*/

//=============================================================================
namespace
{
//-----------------------------------------------------------------------------

//=============================================================================
//PaletteAssignUndo : Undo for the "Load into Current Palette" command.

class PaletteAssignUndo : public TUndo
{
	TPaletteP m_targetPalette, m_oldPalette, m_newPalette;
	TPaletteHandle *m_paletteHandle;

public:
	PaletteAssignUndo(
		const TPaletteP &targetPalette,
		const TPaletteP &oldPalette,
		const TPaletteP &newPalette,
		TPaletteHandle *paletteHandle)
		: m_targetPalette(targetPalette), m_oldPalette(oldPalette), m_newPalette(newPalette), m_paletteHandle(paletteHandle)
	{
	}

	void undo() const
	{
		m_targetPalette->assign(m_oldPalette.getPointer());
		m_paletteHandle->notifyPaletteChanged();
	}
	void redo() const
	{
		m_targetPalette->assign(m_newPalette.getPointer());
		m_paletteHandle->notifyPaletteChanged();
	}

	int getSize() const
	{
		return sizeof(*this) +
			   (m_targetPalette->getStyleCount() +
				m_oldPalette->getStyleCount() +
				m_newPalette->getStyleCount()) *
				   100;
	}

	QString getHistoryString()
	{
		return QObject::tr("Load into Current Palette  > %1")
			.arg(QString::fromStdWString(m_targetPalette->getPaletteName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

//=============================================================================
//StudioPaletteAssignUndo : Undo for the "Replace with Current Palette" command.

class StudioPaletteAssignUndo : public TUndo
{
	TPaletteP m_oldPalette, m_newPalette;
	TFilePath m_fp;
	TPaletteHandle *m_paletteHandle;

public:
	StudioPaletteAssignUndo(
		const TFilePath &targetPath,
		const TPaletteP &oldPalette,
		const TPaletteP &newPalette,
		TPaletteHandle *paletteHandle)
		: m_fp(targetPath), m_oldPalette(oldPalette), m_newPalette(newPalette), m_paletteHandle(paletteHandle)
	{
	}

	void undo() const
	{
		StudioPalette *sp = StudioPalette::instance();
		sp->setPalette(m_fp, m_oldPalette.getPointer(), true);
		m_paletteHandle->notifyPaletteChanged();
	}
	void redo() const
	{
		StudioPalette *sp = StudioPalette::instance();
		sp->setPalette(m_fp, m_newPalette.getPointer(), true);
		m_paletteHandle->notifyPaletteChanged();
	}

	int getSize() const
	{
		return sizeof(*this) +
			   (m_oldPalette->getStyleCount() +
				m_newPalette->getStyleCount()) *
				   100;
	}

	QString getHistoryString()
	{
		return QObject::tr("Replace with Current Palette  > %1")
			.arg(QString::fromStdString(m_fp.getLevelName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

//=============================================================================
//DeletePaletteUndo

class DeletePaletteUndo : public TUndo
{
	TFilePath m_palettePath;
	TPaletteP m_palette;

public:
	DeletePaletteUndo(const TFilePath &palettePath)
		: m_palettePath(palettePath)
	{
		m_palette = StudioPalette::instance()->getPalette(m_palettePath);
	}

	void undo() const
	{
		StudioPalette::instance()->setPalette(m_palettePath, m_palette->clone(), true);
	}
	void redo() const
	{
		StudioPalette::instance()->deletePalette(m_palettePath);
	}
	int getSize() const
	{
		return sizeof(*this) + sizeof(TPalette);
	}
	QString getHistoryString()
	{
		return QObject::tr("Delete Studio Palette  : %1")
			.arg(QString::fromStdString(m_palettePath.getLevelName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

//=============================================================================
//CreatePaletteUndo

class CreatePaletteUndo : public TUndo
{
	TFilePath m_palettePath;
	TPaletteP m_palette;

public:
	CreatePaletteUndo(const TFilePath &palettePath)
		: m_palettePath(palettePath)
	{
		m_palette = StudioPalette::instance()->getPalette(m_palettePath);
	}

	void undo() const
	{
		StudioPalette::instance()->deletePalette(m_palettePath);
	}
	void redo() const
	{
		StudioPalette::instance()->setPalette(m_palettePath, m_palette->clone(), true);
	}
	int getSize() const
	{
		return sizeof(*this) + sizeof(TPalette);
	}
	QString getHistoryString()
	{
		return QObject::tr("Create Studio Palette  : %1")
			.arg(QString::fromStdString(m_palettePath.getLevelName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

//=============================================================================
//DeleteFolderUndo

class DeleteFolderUndo : public TUndo
{
	TFilePath m_path;
	TFilePathSet m_pathSet;
	QList<TPaletteP> m_paletteList;

public:
	DeleteFolderUndo(const TFilePath &path, const TFilePathSet &pathSet)
		: m_path(path), m_pathSet(pathSet), m_paletteList()
	{
		for (TFilePathSet::const_iterator it = m_pathSet.begin(); it != m_pathSet.end(); it++) {
			TFilePath path = *it;
			if (path.getType() == "tpl")
				m_paletteList.push_back(StudioPalette::instance()->getPalette(path));
		}
	}

	void undo() const
	{
		StudioPalette::instance()->createFolder(m_path.getParentDir(), m_path.getWideName());
		int paletteCount = -1;
		for (TFilePathSet::const_iterator it = m_pathSet.begin(); it != m_pathSet.end(); it++) {
			TFilePath path = *it;
			if (path.getType() == "tpl") //Is a palette
				StudioPalette::instance()->setPalette(path, m_paletteList.at(++paletteCount)->clone(), true);
			else //Is a folder
				StudioPalette::instance()->createFolder(path.getParentDir(), path.getWideName());
		}
	}
	void redo() const
	{
		StudioPalette::instance()->deleteFolder(m_path);
	}
	int getSize() const
	{
		return sizeof(*this) + sizeof(TPalette);
	}
	QString getHistoryString()
	{
		return QObject::tr("Delete Studio Palette Folder  : %1")
			.arg(QString::fromStdString(m_path.getName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

//=============================================================================
//CreateFolderUndo

class CreateFolderUndo : public TUndo
{
	TFilePath m_folderPath;

public:
	CreateFolderUndo(const TFilePath &folderPath)
		: m_folderPath(folderPath)
	{
	}

	void undo() const
	{
		StudioPalette::instance()->deleteFolder(m_folderPath);
	}
	void redo() const
	{
		StudioPalette::instance()->createFolder(m_folderPath.getParentDir(), m_folderPath.getWideName());
	}
	int getSize() const
	{
		return sizeof(*this) + sizeof(TPalette);
	}
	QString getHistoryString()
	{
		return QObject::tr("Create Studio Palette Folder  : %1")
			.arg(QString::fromStdString(m_folderPath.getName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

//=============================================================================
//MovePaletteUndo

class MovePaletteUndo : public TUndo
{
	TFilePath m_dstPath, m_srcPath;

public:
	MovePaletteUndo(const TFilePath &dstPath, const TFilePath &srcPath)
		: m_dstPath(dstPath), m_srcPath(srcPath)
	{
	}

	void undo() const
	{
		StudioPalette::instance()->movePalette(m_srcPath, m_dstPath);
	}
	void redo() const
	{
		StudioPalette::instance()->movePalette(m_dstPath, m_srcPath);
	}
	int getSize() const
	{
		return sizeof(*this);
	}
	QString getHistoryString()
	{
		return QObject::tr("Move Studio Palette Folder  : %1 : %2 > %3")
			.arg(QString::fromStdString(m_srcPath.getName()))
			.arg(QString::fromStdString(m_srcPath.getParentDir().getName()))
			.arg(QString::fromStdString(m_dstPath.getParentDir().getName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

//-------------------------------------------------------------
void adaptLevelToPalette(TXshLevelHandle *currentLevelHandle, TPaletteHandle *paletteHandle, TPalette *plt, int tolerance, bool noUndo);

class AdjustIntoCurrentPaletteUndo : public TUndo
{
	TXshLevelHandle *m_currentLevelHandle;
	TPaletteHandle *m_paletteHandle;
	TPaletteP m_oldPalette, m_newPalette;
	TFrameId m_fid;
	std::string m_oldImageId;
	static int m_idCount;
	int m_undoSize;
	int m_tolerance;

public:
	AdjustIntoCurrentPaletteUndo(TXshLevelHandle *currentLevelHandle, TPaletteHandle *paletteHandle, const TFrameId &fid,
								 const TImageP &oldImage, const TPaletteP &oldPalette, const TPaletteP &newPalette, int tolerance)
		: TUndo(), m_currentLevelHandle(currentLevelHandle), m_paletteHandle(paletteHandle), m_fid(fid), m_oldPalette(oldPalette), m_newPalette(newPalette), m_tolerance(tolerance)
	{
		m_oldImageId = "AdjustIntoCurrentPaletteUndo_oldImage_" + std::to_string(m_idCount++);

		TImageCache::instance()->add(m_oldImageId, (const TImageP)oldImage);

		m_undoSize = ((TToonzImageP)oldImage)->getRaster()->getLx() * ((TToonzImageP)oldImage)->getRaster()->getLy() * ((TToonzImageP)oldImage)->getRaster()->getPixelSize();
	}

	~AdjustIntoCurrentPaletteUndo()
	{
		TImageCache::instance()->remove(m_oldImageId);
	}

	void undo() const
	{

		TImageP img = TImageCache::instance()->get(m_oldImageId, true);
		TXshSimpleLevel *level = m_currentLevelHandle->getSimpleLevel();
		level->setPalette(m_oldPalette.getPointer());
		//img->setPalette(m_oldPalette.getPointer());
		level->setFrame(m_fid, img->cloneImage());

		level->touchFrame(m_fid);
		if (level->getFirstFid() == m_fid) {
			m_currentLevelHandle->notifyLevelChange();
			m_paletteHandle->setPalette(m_oldPalette.getPointer());

			m_oldPalette->setDirtyFlag(true);
			m_paletteHandle->notifyPaletteChanged();
		}
	}

	void redo() const
	{
		adaptLevelToPalette(m_currentLevelHandle, m_paletteHandle, m_newPalette.getPointer(), m_tolerance, true);
	}

	int getSize() const
	{
		return m_undoSize;
	}
};

//-----------------------------------------------------------------------------
int AdjustIntoCurrentPaletteUndo::m_idCount = 0;

//-----------------------------------------------------------------------------
} // namespace
//=============================================================================

void StudioPaletteCmd::loadIntoCurrentPalette(TPaletteHandle *paletteHandle,
											  const TFilePath &fp)
{
	TPalette *palette = StudioPalette::instance()->getPalette(fp, false);
	if (!palette)
		return;

	loadIntoCurrentPalette(paletteHandle, palette);
}

//-----------------------------------------------------------------------------
namespace
{

std::map<TPixel, int> ToleranceMap;

int findClosest(const TPixel &color, std::map<TPixel, int> &colorMap, int tolerance)
{
	std::map<TPixel, int>::const_iterator it;
	it = ToleranceMap.find(color);
	if (it != ToleranceMap.end())
		return it->second;

	tolerance *= tolerance;

	it = colorMap.begin();
	int index = -1, minDist = 99999999;
	int dr, dg, db, dm;

	for (; it != colorMap.end(); ++it) {
		if ((dr = (color.r - it->first.r) * (color.r - it->first.r)) > tolerance)
			continue;
		if ((dg = (color.g - it->first.g) * (color.g - it->first.g)) > tolerance)
			continue;
		if ((db = (color.b - it->first.b) * (color.b - it->first.b)) > tolerance)
			continue;
		if ((dm = (color.m - it->first.m) * (color.m - it->first.m)) > tolerance)
			continue;

		int currDist = dr + dg + db + dm;
		if (currDist < minDist)
			minDist = currDist, index = it->second;
	}
	ToleranceMap[color] = index;
	return index;
}

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

int getIndex(const TPixel &color, std::map<TPixel, int> &colorMap, TPalette *plt, int tolerance)
{
	std::map<TPixel, int>::const_iterator it;
	it = colorMap.find(color);
	if (it != colorMap.end())
		return it->second;

	if (tolerance > 0) {
		int index = findClosest(color, colorMap, tolerance);
		if (index != -1)
			return index;
	}

	int index = plt->addStyle(color);
	plt->getPage(0)->addStyle(index);
	colorMap[color] = index;
	return index;
}

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

void adaptIndexes(TToonzImageP timg, std::map<TPixel, int> &colorMap, TPalette *plt, int tolerance)
{
	TPalette *origPlt = timg->getPalette();
	TRasterCM32P r = timg->getRaster();
	TPixelCM32 oldPrev = TPixelCM32(), newPrev = TPixelCM32();

	for (int i = 0; i < r->getLy(); i++) {
		TPixelCM32 *pix = r->pixels(i);
		for (int j = 0; j < r->getLx(); j++, pix++) {
			if (*pix == TPixelCM32())
				continue;
			if (*pix == oldPrev) {
				*pix = newPrev;
				continue;
			}
			oldPrev = *pix;
			int ink = pix->getInk();
			int paint = pix->getPaint();
			if (ink > 0) {
				TPixel color = origPlt->getStyle(ink)->getMainColor();
				int newIndex = getIndex(color, colorMap, plt, tolerance);
				pix->setInk(newIndex);
			}
			if (paint > 0) {
				TPixel color = origPlt->getStyle(paint)->getMainColor();
				int newIndex = getIndex(color, colorMap, plt, tolerance);
				pix->setPaint(newIndex);
			}
			newPrev = *pix;
		}
	}
}

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

void adaptLevelToPalette(TXshLevelHandle *currentLevelHandle, TPaletteHandle *paletteHandle, TPalette *plt, int tolerance, bool noUndo)
{
	TXshSimpleLevel *sl = currentLevelHandle->getSimpleLevel();
	QApplication::setOverrideCursor(Qt::WaitCursor);

	TPalette *oldPalette = sl->getPalette();

	ToleranceMap.clear();

	std::map<TPixel, int> colorMap;
	for (int i = 0; i < plt->getStyleCount(); i++) {
		if (!plt->getStylePage(i))
			continue;
		colorMap[plt->getStyle(i)->getMainColor()] = i;
	}
	std::vector<TFrameId> fids;

	sl->getFids(fids);
	for (int i = 0; i < (int)fids.size(); i++) {
		TToonzImageP timg = (TToonzImageP)sl->getFrame(fids[i], true);
		if (!timg)
			continue;
		if (!noUndo)
			TUndoManager::manager()->add(
				new AdjustIntoCurrentPaletteUndo(currentLevelHandle, paletteHandle, fids[i], timg->cloneImage(), oldPalette->clone(), plt->clone(), tolerance));

		adaptIndexes(timg, colorMap, plt, tolerance);
		timg->setPalette(plt);
	}
	QApplication::restoreOverrideCursor();

	currentLevelHandle->getSimpleLevel()->setPalette(plt);
	paletteHandle->setPalette(plt);
	plt->setDirtyFlag(true);
	paletteHandle->notifyPaletteChanged();
	currentLevelHandle->notifyLevelChange();
}

} //namespaxce

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

void StudioPaletteCmd::loadIntoCurrentPalette(TPaletteHandle *paletteHandle, TPalette *plt, TXshLevelHandle *currentLevelHandle, int tolerance)
{
	adaptLevelToPalette(currentLevelHandle, paletteHandle, plt, tolerance, false);
}

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

void StudioPaletteCmd::loadIntoCurrentPalette(TPaletteHandle *paletteHandle,
											  TPalette *palette)
{
	assert(paletteHandle);
	TPalette *current = paletteHandle->getPalette();
	if (!current)
		return;

	int styleId = paletteHandle->getStyleIndex();

	TPalette *old = current->clone();
	while (palette->getStyleCount() < current->getStyleCount()) {
		int index = palette->getStyleCount();
		assert(index < current->getStyleCount());
		TColorStyle *style = current->getStyle(index)->clone();
		palette->addStyle(style);
	}
	//keep the color model path unchanged
	TFilePath oldRefImagePath = current->getRefImgPath();

	current->assign(palette, true);
	current->setDirtyFlag(true);

	current->setRefImgPath(oldRefImagePath);

	if (paletteHandle->getPalette() == current && styleId >= current->getStyleCount())
		paletteHandle->setStyleIndex(1);
	TUndoManager::manager()->add(
		new PaletteAssignUndo(current, old, current->clone(), paletteHandle));

	palette->setDirtyFlag(true);

	paletteHandle->notifyPaletteChanged();
}

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

void StudioPaletteCmd::mergeIntoCurrentPalette(TPaletteHandle *paletteHandle,
											   const TFilePath &fp)
{
	TPalette *palette = StudioPalette::instance()->getPalette(fp);

	if (!palette)
		return;

	mergeIntoCurrentPalette(paletteHandle, palette);
}

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

void StudioPaletteCmd::mergeIntoCurrentPalette(TPaletteHandle *paletteHandle,
											   TPalette *palette)
{
	assert(paletteHandle);
	TPalette *current = paletteHandle->getPalette();

	if (!current)
		return;
	if (current->isLocked())
		return;

	TPalette *old = current->clone();
	current->merge(palette, true);
	TUndoManager::manager()->add(
		new PaletteAssignUndo(current, old, current->clone(), paletteHandle));

	palette->setDirtyFlag(true);
	paletteHandle->notifyPaletteChanged();
}

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

void StudioPaletteCmd::replaceWithCurrentPalette(TPaletteHandle *paletteHandle,
												 TPaletteHandle *stdPaletteHandle,
												 const TFilePath &fp)
{
	StudioPalette *sp = StudioPalette::instance();
	TPalette *palette = sp->getPalette(fp);

	if (!palette || palette->isLocked())
		return;

	assert(paletteHandle);
	TPalette *current = paletteHandle->getPalette();

	if (!current)
		return;
	//keep the studio palette's global name unchanged
	std::wstring oldGlobalName = palette->getGlobalName();

	TPalette *old = palette->clone();
	palette->assign(current);

	//put back the global name
	palette->setGlobalName(oldGlobalName);

	sp->setPalette(fp, current, true);
	TUndoManager::manager()->add(
		new StudioPaletteAssignUndo(fp, old, current->clone(), paletteHandle));

	// Cambio la studioPalette correntette(palette);
	stdPaletteHandle->setPalette(palette);

	stdPaletteHandle->notifyPaletteSwitched();
}

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

void StudioPaletteCmd::updateAllLinkedStyles(TPaletteHandle *paletteHandle, TXsheetHandle *xsheetHandle)
{
	if (!xsheetHandle)
		return;
	TXsheet *xsheet = xsheetHandle->getXsheet();
	if (!xsheet)
		return;
	ToonzScene *scene = xsheet->getScene();
	if (!scene)
		return;

	//emit signal only if something changed
	bool somethingChanged = false;

	StudioPalette *sp = StudioPalette::instance();
	TLevelSet *levelSet = scene->getLevelSet();
	for (int i = 0; i < levelSet->getLevelCount(); i++) {
		TXshLevel *xl = levelSet->getLevel(i);
		TXshSimpleLevel *sl = xl ? xl->getSimpleLevel() : 0;
		if (!sl)
			continue;
		TPalette *palette = sl->getPalette();
		if (palette) {
			somethingChanged = somethingChanged | sp->updateLinkedColors(palette);
			if (sl->getType() == TZP_XSHLEVEL) {
				std::vector<TFrameId> fids;
				sl->getFids(fids);
				std::vector<TFrameId>::iterator it;
				for (it = fids.begin(); it != fids.end(); ++it) {
					TFrameId fid = *it;
					std::string id = sl->getImageId(fid);
				}
			}
		}
	}
	if (!paletteHandle || !paletteHandle->getPalette())
		return;
	if (somethingChanged)
		paletteHandle->notifyColorStyleChanged();
}

//-----------------------------------------------------------------------------
/*! Delete palette identified by \b fp in \b StudioPalette. If there are any
		problems in loading send an error message.
*/
void StudioPaletteCmd::deletePalette(const TFilePath &fp)
{
	TUndo *undo = new DeletePaletteUndo(fp);
	StudioPalette::instance()->deletePalette(fp);
	TUndoManager::manager()->add(undo);
}

//-----------------------------------------------------------------------------
/*! Move palette from TFilePath \b srcPath to \b TFilePath \b dstPath. If there
		are any problems in moving send an error message.
*/
void StudioPaletteCmd::movePalette(const TFilePath &dstPath, const TFilePath &srcPath)
{
	TSystem::touchParentDir(dstPath);
	StudioPalette::instance()->movePalette(dstPath, srcPath);
	TUndoManager::manager()->add(new MovePaletteUndo(dstPath, srcPath));
}

//-----------------------------------------------------------------------------
/*! Create palette \b palette in folder \b folderName with name \b paletteName.
		If there are any problems send an error message.
*/
TFilePath StudioPaletteCmd::createPalette(const TFilePath &folderName,
										  std::string paletteName,
										  const TPalette *palette)
{
	TFilePath palettePath;
	TFileStatus status(folderName);
	if (!status.isDirectory())
		throw TException("Select a folder.");
	if (!status.doesExist()) {
		TSystem::mkDir(folderName);
		FolderListenerManager::instance()->notifyFolderChanged(folderName.getParentDir());
	}
	palettePath = StudioPalette::instance()->createPalette(folderName, paletteName);
	if (palette)
		StudioPalette::instance()->setPalette(palettePath, palette, true);
	TUndoManager::manager()->add(new CreatePaletteUndo(palettePath));
	return palettePath;
}

//-----------------------------------------------------------------------------
/*! Add a folder in StudioPalette TFilePath \b parentFolderPath. If there
		are any problems send an error message.
*/
TFilePath StudioPaletteCmd::addFolder(const TFilePath &parentFolderPath)
{
	TFilePath folderPath;
	folderPath = StudioPalette::instance()->createFolder(parentFolderPath);
	if (!folderPath.isEmpty())
		TUndoManager::manager()->add(new CreateFolderUndo(folderPath));
	return folderPath;
}

//-----------------------------------------------------------------------------
/*! Delete folder \b folderPath. If there are any problems in deleting send an
		error message.
*/
void StudioPaletteCmd::deleteFolder(const TFilePath &folderPath)
{
	TFilePathSet folderPathSet;
	TSystem::readDirectoryTree(folderPathSet, folderPath, true, false);
	TUndo *undo = new DeleteFolderUndo(folderPath, folderPathSet);
	StudioPalette::instance()->deleteFolder(folderPath);
	TUndoManager::manager()->add(undo);
}

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

void StudioPaletteCmd::scanPalettes(const TFilePath &folder, const TFilePath &sourcePath)
{
	//  error("uh oh");
}