Blob Blame Raw


#include "toonz/palettecmd.h"

// TnzLib includes
#include "toonz/tpalettehandle.h"
#include "toonz/txshlevelhandle.h"
#include "toonz/txsheethandle.h"
#include "toonz/studiopalette.h"
#include "toonz/txsheet.h"
#include "toonz/txshcolumn.h"
#include "toonz/txshcell.h"
#include "toonz/txshlevelcolumn.h"
#include "toonz/txshsimplelevel.h"
#include "toonz/cleanupcolorstyles.h"
#include "toonz/studiopalette.h"
#include "toonz/txshlevel.h"
#include "toonz/toonzscene.h"
#include "toonz/toonzimageutils.h"
#include "toonz/cleanupcolorstyles.h"
#include "toonz/preferences.h"

// TnzCore includes
#include "tpalette.h"
#include "tcolorstyles.h"
#include "tundo.h"
#include "tvectorimage.h"
#include "ttoonzimage.h"
#include "trasterimage.h"
#include "tconvert.h"
#include "tcolorutils.h"
#include "tropcm.h"
#include "tstroke.h"
#include "tregion.h"
#include "tlevel_io.h"

#include "historytypes.h"

// tcg includes
#include "tcg/boost/range_utility.h"

// boost includes
#include <boost/bind.hpp>
#include <boost/range/counting_range.hpp>
#include <boost/range/adaptor/transformed.hpp>
#include <boost/range/adaptor/filtered.hpp>

//===================================================================

void findPaletteLevels(set<TXshSimpleLevel *> &levels, int &rowIndex, int &columnIndex,
					   TPalette *palette, TXsheet *xsheet)
{
	rowIndex = columnIndex = -1;
	int columnCount = xsheet->getColumnCount();
	int c;
	for (c = 0; c < columnCount; c++) {
		TXshColumn *column = xsheet->getColumn(c);
		if (!column || column->isEmpty())
			continue;

		TXshLevelColumn *levelColumn = column->getLevelColumn();
		if (!levelColumn || levelColumn->isEmpty())
			continue;

		int r0, r1;
		if (!column->getRange(r0, r1))
			continue;
		int r;
		for (r = r0; r <= r1; r++) {
			TXshCell cell = levelColumn->getCell(r);
			if (cell.isEmpty())
				continue;
			TXshSimpleLevel *level = cell.getSimpleLevel();
			if (!level || level->getPalette() != palette)
				continue;
			levels.insert(level);
			if (rowIndex < 0) {
				rowIndex = r;
				columnIndex = c;
			}
		}
	}
}

//===================================================================

namespace
{

bool isStyleUsed(const TVectorImageP vi, int styleId)
{
	int strokeCount = vi->getStrokeCount();
	int i;
	for (i = strokeCount - 1; i >= 0; i--) {
		TStroke *stroke = vi->getStroke(i);
		if (stroke && stroke->getStyle() == styleId)
			return true;
	}
	int regionCount = vi->getRegionCount();
	for (i = 0; i < regionCount; i++) {
		TRegion *region = vi->getRegion(i);
		if (region || region->getStyle() != styleId)
			return true;
	}
	return false;
}

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

bool isStyleUsed(const TToonzImageP vi, int styleId)
{
	TRasterCM32P ras = vi->getRaster();
	for (int y = 0; y < ras->getLy(); y++) {
		TPixelCM32 *pix = ras->pixels(y), *endPix = pix + ras->getLx();
		while (pix < endPix) {
			if (pix->getPaint() == styleId)
				return true;
			if (pix->getInk() == styleId)
				return true;
			pix++;
		}
	}
	return false;
}

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

/*! Return true if one style is used. */
bool areStylesUsed(const TImageP image, const vector<int> styleIds)
{
	int j;
	for (j = 0; j < (int)styleIds.size(); j++)
		if (isStyleUsed(image, styleIds[j]))
			return true;
	return false;
}

} //namespace

//===================================================================

bool isStyleUsed(const TImageP image, int styleId)
{
	TVectorImageP vi = image;
	TToonzImageP ti = image;
	if (vi)
		return isStyleUsed(vi, styleId);
	if (ti)
		return isStyleUsed(ti, styleId);
	return false;
}

//===================================================================

/*! Return true if one style is used. */
bool areStylesUsed(const set<TXshSimpleLevel *> levels, const vector<int> styleIds)
{
	set<TXshSimpleLevel *>::const_iterator it = levels.begin();
	for (it; it != levels.end(); it++) {
		TXshSimpleLevel *level = *it;
		vector<TFrameId> fids;
		level->getFids(fids);
		int i;
		for (i = 0; i < (int)fids.size(); i++) {
			TImageP image = level->getFrame(fids[i], true);
			if (areStylesUsed(image, styleIds))
				return true;
		}
	}
	return false;
}

//===================================================================
//
// arrangeStyles
// srcPage : {a0,a1,...an } ==> dstPage : {b,b+1,...b+n-1}
//
//-------------------------------------------------------------------

/*! \namespace PaletteCmd
		\brief Provides a collection of methods to manage \b TPalette.
*/

namespace
{

class ArrangeStylesUndo : public TUndo
{
	TPaletteHandle *m_paletteHandle;
	TPaletteP m_palette;
	int m_dstPageIndex;
	int m_dstIndexInPage;
	int m_srcPageIndex;
	std::set<int> m_srcIndicesInPage;

public:
	ArrangeStylesUndo(TPaletteHandle *paletteHandle,
					  int dstPageIndex,
					  int dstIndexInPage,
					  int srcPageIndex,
					  const std::set<int> &srcIndicesInPage)
		: m_paletteHandle(paletteHandle), m_dstPageIndex(dstPageIndex), m_dstIndexInPage(dstIndexInPage), m_srcPageIndex(srcPageIndex), m_srcIndicesInPage(srcIndicesInPage)
	{
		m_palette = m_paletteHandle->getPalette();
		assert(m_palette);
		assert(0 <= dstPageIndex && dstPageIndex < m_palette->getPageCount());
		assert(0 <= srcPageIndex && srcPageIndex < m_palette->getPageCount());
		TPalette::Page *dstPage = m_palette->getPage(dstPageIndex);
		assert(dstPage);
		assert(0 <= dstIndexInPage && dstIndexInPage <= dstPage->getStyleCount());
		assert(!srcIndicesInPage.empty());
		TPalette::Page *srcPage = m_palette->getPage(srcPageIndex);
		assert(srcPage);
		assert(0 <= *srcIndicesInPage.begin() && *srcIndicesInPage.rbegin() < srcPage->getStyleCount());
	}
	void undo() const
	{
		TPalette::Page *srcPage = m_palette->getPage(m_srcPageIndex);
		assert(srcPage);
		TPalette::Page *dstPage = m_palette->getPage(m_dstPageIndex);
		assert(dstPage);
		std::vector<int> styles;
		int count = m_srcIndicesInPage.size();
		int h = m_dstIndexInPage;
		std::set<int>::const_iterator i;
		for (i = m_srcIndicesInPage.begin(); i != m_srcIndicesInPage.end(); ++i) {
			if (srcPage == dstPage && (*i) <= h)
				h--;
			else
				break;
		}
		assert(h + count - 1 <= dstPage->getStyleCount());
		int k;
		for (k = 0; k < count; k++) {
			styles.push_back(dstPage->getStyleId(h));
			dstPage->removeStyle(h);
		}
		k = 0;
		for (i = m_srcIndicesInPage.begin(); i != m_srcIndicesInPage.end(); ++i, ++k)
			srcPage->insertStyle(*i, styles[k]);

		m_paletteHandle->notifyPaletteChanged();
	}
	void redo() const
	{
		TPalette::Page *srcPage = m_palette->getPage(m_srcPageIndex);
		assert(srcPage);
		TPalette::Page *dstPage = m_palette->getPage(m_dstPageIndex);
		assert(dstPage);

		std::vector<int> styles;
		std::set<int>::const_reverse_iterator i;
		std::vector<int>::iterator j;
		int k = m_dstIndexInPage;
		for (i = m_srcIndicesInPage.rbegin(); i != m_srcIndicesInPage.rend(); ++i) {
			int index = *i;
			if (m_dstPageIndex == m_srcPageIndex && index < k)
				k--;
			styles.push_back(srcPage->getStyleId(index));
			srcPage->removeStyle(index);
		}
		for (j = styles.begin(); j != styles.end(); ++j)
			dstPage->insertStyle(k, *j);
		m_palette->setDirtyFlag(true);
		m_paletteHandle->notifyPaletteChanged();
	}

	int getSize() const { return sizeof *this + m_srcIndicesInPage.size() * sizeof(int); }

	QString getHistoryString()
	{
		return QObject::tr("Arrange Styles  in Palette %1")
			.arg(QString::fromStdWString(m_palette->getPaletteName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

} // namespace

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

void PaletteCmd::arrangeStyles(TPaletteHandle *paletteHandle,
							   int dstPageIndex, int dstIndexInPage,
							   int srcPageIndex, const std::set<int> &srcIndicesInPage)
{
	if (dstPageIndex == srcPageIndex && dstIndexInPage == *srcIndicesInPage.begin())
		return;
	ArrangeStylesUndo *undo =
		new ArrangeStylesUndo(paletteHandle, dstPageIndex,
							  dstIndexInPage, srcPageIndex,
							  srcIndicesInPage);
	undo->redo();
	TUndoManager::manager()->add(undo);
}

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

class CreateStyleUndo : public TUndo
{
	TPaletteHandle *m_paletteHandle;
	TPaletteP m_palette;
	int m_pageIndex;
	int m_styleId;
	TColorStyle *m_style;

public:
	CreateStyleUndo(
		TPaletteHandle *paletteHandle,
		int pageIndex,
		int styleId)
		: m_paletteHandle(paletteHandle), m_pageIndex(pageIndex), m_styleId(styleId)
	{
		m_palette = m_paletteHandle->getPalette();
		m_style = m_palette->getStyle(m_styleId)->clone();
		assert(m_palette);
		assert(0 <= pageIndex && pageIndex < m_palette->getPageCount());
		assert(0 <= styleId && styleId < m_palette->getStyleCount());
	}
	void undo() const
	{
		TPalette::Page *page = m_palette->getPage(m_pageIndex);
		assert(page);
		int indexInPage = page->search(m_styleId);
		assert(indexInPage >= 0);
		page->removeStyle(indexInPage);
		m_paletteHandle->notifyPaletteChanged();
	}
	void redo() const
	{
		TPalette::Page *page = m_palette->getPage(m_pageIndex);
		assert(page);
		assert(m_palette->getStylePage(m_styleId) == 0);
		int indexInPage = page->addStyle(m_styleId);
		if (indexInPage == -1) {
			indexInPage = page->getStyleCount();
			page->insertStyle(indexInPage, m_style->getMainColor());
			assert(m_styleId == page->getStyleId(indexInPage));
		}
		m_palette->getStyle(m_styleId)->setMainColor(m_style->getMainColor());
		m_palette->getStyle(m_styleId)->setName(m_style->getName());
		m_paletteHandle->notifyPaletteChanged();
	}
	int getSize() const { return sizeof *this; }
	QString getHistoryString()
	{
		return QObject::tr("Create Style#%1  in Palette %2")
			.arg(QString::number(m_styleId))
			.arg(QString::fromStdWString(m_palette->getPaletteName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

} // namespace

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

void PaletteCmd::createStyle(
	TPaletteHandle *paletteHandle,
	TPalette::Page *page)
{
	int index = paletteHandle->getStyleIndex();
	TPalette *palette = paletteHandle->getPalette();
	int newIndex;

	int UnpagedId = palette->getFirstUnpagedStyle();
	if (UnpagedId != -1 && !palette->isCleanupPalette()) {
		if (index == -1)
			palette->getStyle(UnpagedId)->setMainColor(TPixel32::Black);
		else
			palette->getStyle(UnpagedId)->setMainColor(palette->getStyle(index)->getMainColor());
		newIndex = page->addStyle(UnpagedId);
	} else if (!palette->isCleanupPalette()) {
		if (index == -1)
			newIndex = page->addStyle(TPixel32::Black);
		else {
			TColorStyle *style = palette->getStyle(index);
			TCleanupStyle *cleanupStyle = dynamic_cast<TCleanupStyle *>(style);
			if ((cleanupStyle || index == 0) && palette->isCleanupPalette()) {
				TColorCleanupStyle *newCleanupStyle = new TColorCleanupStyle();
				if (cleanupStyle) {
					int i;
					for (i = 0; i < cleanupStyle->getParamCount(); i++)
						newCleanupStyle->setColorParamValue(i, cleanupStyle->getColorParamValue(i));
				}
				newIndex = page->addStyle(newCleanupStyle);
			} else
				newIndex = page->addStyle(style->getMainColor());
		}
	} else /*- CleanupPaletteの場合 -*/
	{
		newIndex = page->addStyle(new TColorCleanupStyle(TPixel32::Red));
	}
	int newStyleId = page->getStyleId(newIndex);
	/*-  StudioPalette上でStyleを追加した場合、GlobalNameを自動で割り振る -*/
	if (palette->getGlobalName() != L"") {
		TColorStyle *cs = palette->getStyle(newStyleId);
		wstring gname = L"-" + palette->getGlobalName() + L"-" + toWideString(newStyleId);
		cs->setGlobalName(gname);
	}

	page->getStyle(newIndex)->setName(QString("color_%1").arg(newStyleId).toStdWString());
	paletteHandle->setStyleIndex(newStyleId);

	palette->setDirtyFlag(true);
	paletteHandle->notifyPaletteChanged();
	TUndoManager::manager()->add(new CreateStyleUndo(paletteHandle, page->getIndex(), page->getStyleId(newIndex)));
}

//=============================================================================
// addStyles
//-----------------------------------------------------------------------------

namespace
{

//=============================================================================
// AddStylesUndo
//-----------------------------------------------------------------------------

class AddStylesUndo : public TUndo
{
	TPaletteP m_palette;
	int m_pageIndex;
	int m_indexInPage;
	std::vector<std::pair<TColorStyle *, int>> m_styles;
	TPaletteHandle *m_paletteHandle;

public:
	// creare DOPO l'inserimento
	AddStylesUndo(
		const TPaletteP &palette,
		int pageIndex,
		int indexInPage,
		int count,
		TPaletteHandle *paletteHandle)
		: m_palette(palette), m_pageIndex(pageIndex), m_indexInPage(indexInPage), m_paletteHandle(paletteHandle)
	{
		assert(m_palette);
		assert(0 <= m_pageIndex && m_pageIndex < m_palette->getPageCount());
		TPalette::Page *page = m_palette->getPage(m_pageIndex);
		assert(page);
		assert(0 <= indexInPage && indexInPage + count <= page->getStyleCount());
		for (int i = 0; i < count; i++) {
			std::pair<TColorStyle *, int> p;
			p.second = page->getStyleId(m_indexInPage + i);
			p.first = m_palette->getStyle(p.second)->clone();
			m_styles.push_back(p);
		}
	}
	~AddStylesUndo()
	{
		for (int i = 0; i < (int)m_styles.size(); i++)
			delete m_styles[i].first;
	}
	void undo() const
	{
		TPalette::Page *page = m_palette->getPage(m_pageIndex);
		assert(page);
		int count = m_styles.size();
		int i;
		for (i = count - 1; i >= 0; i--)
			page->removeStyle(m_indexInPage + i);
		m_paletteHandle->notifyPaletteChanged();
	}
	void redo() const
	{
		TPalette::Page *page = m_palette->getPage(m_pageIndex);
		assert(page);
		for (int i = 0; i < (int)m_styles.size(); i++) {
			TColorStyle *cs = m_styles[i].first->clone();
			int styleId = m_styles[i].second;
			assert(m_palette->getStylePage(styleId) == 0);
			m_palette->setStyle(styleId, cs);
			page->insertStyle(m_indexInPage + i, styleId);
		}
		m_paletteHandle->notifyPaletteChanged();
	}
	int getSize() const { return sizeof *this + m_styles.size() * sizeof(TColorStyle); }

	QString getHistoryString()
	{
		return QObject::tr("Add Style  to Palette %1")
			.arg(QString::fromStdWString(m_palette->getPaletteName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

} // namespace

//-----------------------------------------------------------------------------
/*- ドラッグ&ドロップ時に呼ばれる -*/
void PaletteCmd::addStyles(TPaletteHandle *paletteHandle, int pageIndex,
						   int indexInPage, const std::vector<TColorStyle *> &styles)
{
	TPalette *palette = paletteHandle->getPalette();
	assert(0 <= pageIndex && pageIndex < palette->getPageCount());
	TPalette::Page *page = palette->getPage(pageIndex);
	assert(page);
	assert(0 <= indexInPage && indexInPage <= page->getStyleCount());
	int count = styles.size();
	for (int i = 0; i < count; i++) {
		page->insertStyle(indexInPage + i, styles[i]->clone());

		/*-- StudioPaletteから持ち込んだ場合、その元の名前をoriginalNameに入れる --*/
		if (styles[i]->getGlobalName() != L"") {
			/*-- もし元のStyleにOriginalNameが無ければ(コピー元がStudioPaletteからということ)--*/
			if (styles[i]->getOriginalName() == L"") {
				/*-- 元のStyleの名前をペースト先のOriginalNameに入れる --*/
				page->getStyle(indexInPage + i)->setOriginalName(styles[i]->getName());
			}
		}
		/*-- それ以外の場合は、clone()でそれぞれの名前をコピーしているので、そのままでOK --*/
	}
	TUndoManager::manager()->add(new AddStylesUndo(palette, pageIndex, indexInPage, count, paletteHandle));
	palette->setDirtyFlag(true);
}

//=============================================================================
// eraseStyles
//-----------------------------------------------------------------------------

namespace
{

void eraseStylesInLevels(const set<TXshSimpleLevel *> &levels, const vector<int> styleIds)
{
	set<TXshSimpleLevel *>::const_iterator it = levels.begin();
	for (it; it != levels.end(); it++) {
		TXshSimpleLevel *level = *it;
		vector<TFrameId> fids;
		level->getFids(fids);
		int i;
		for (i = 0; i < (int)fids.size(); i++) {
			TImageP image = level->getFrame(fids[i], true);
			TVectorImageP vi = image;
			TToonzImageP ti = image;
			if (vi)
				vi->eraseStyleIds(styleIds);
			else if (ti)
				TRop::eraseStyleIds(ti.getPointer(), styleIds);
		}
	}
}

} // namespace

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

void PaletteCmd::eraseStyles(const std::set<TXshSimpleLevel *> &levels,
							 const std::vector<int> &styleIds)
{
	typedef std::pair<const TXshSimpleLevelP, std::vector<TVectorImageP>> LevelImages;

	struct Undo : public TUndo {
		std::set<TXshSimpleLevel *> m_levels;
		std::vector<int> m_styleIds;

		mutable std::map<TXshSimpleLevelP, std::vector<TVectorImageP>> m_imagesByLevel;

	public:
		Undo(const std::set<TXshSimpleLevel *> &levels, const std::vector<int> &styleIds)
			: m_levels(levels), m_styleIds(styleIds)
		{
			tcg::substitute(m_imagesByLevel, levels | boost::adaptors::filtered(isVector) | boost::adaptors::transformed(toEmptyLevelImages));
		}

		bool isValid() const { return !m_levels.empty(); }

		void redo() const
		{
			std::for_each(m_imagesByLevel.begin(), m_imagesByLevel.end(), cloneImages);
			eraseStylesInLevels(m_levels, m_styleIds);
		}

		void undo() const
		{
			std::for_each(m_imagesByLevel.begin(), m_imagesByLevel.end(), restoreImages);
		}

		int getSize() const { return 10 << 20; } // At max 10 per 100 MB

	private:
		static bool isVector(const TXshSimpleLevel *level)
		{
			return (assert(level), level->getType() == PLI_XSHLEVEL);
		}

		static LevelImages toEmptyLevelImages(TXshSimpleLevel *level)
		{
			return LevelImages(level, std::vector<TVectorImageP>());
		}

		static void copyStrokeIds(const TVectorImage &src, TVectorImage &dst)
		{
			UINT s, sCount = src.getStrokeCount();
			for (s = 0; s != sCount; ++s)
				dst.getStroke(s)->setId(src.getStroke(s)->getId());
		}

		static TVectorImageP cloneImage(const TXshSimpleLevel &level, int f)
		{
			TVectorImageP src = static_cast<TVectorImageP>(level.getFrame(level.getFrameId(f), false));
			TVectorImageP dst = src->clone();

			copyStrokeIds(*src, *dst);
			return dst;
		}

		static void cloneImages(LevelImages &levelImages)
		{
			tcg::substitute(levelImages.second, boost::counting_range(0, levelImages.first->getFrameCount()) | boost::adaptors::transformed(boost::bind(cloneImage, boost::cref(*levelImages.first), _1)));
		}

		static void restoreImage(const TXshSimpleLevelP &level, int f, const TVectorImageP &vi)
		{
			level->setFrame(level->getFrameId(f), vi.getPointer());
		}

		static void restoreImages(LevelImages &levelImages)
		{
			int f, fCount = tmin(levelImages.first->getFrameCount(), int(levelImages.second.size()));

			for (f = 0; f != fCount; ++f)
				restoreImage(levelImages.first, f, levelImages.second[f]);
		}

	}; // Undo

	if (levels.empty() || styleIds.empty())
		return;

	std::auto_ptr<TUndo> undo(new Undo(levels, styleIds));
	if (static_cast<Undo &>(*undo).isValid()) {
		undo->redo();
		TUndoManager::manager()->add(undo.release());
	}
}

//=============================================================================
// addPage
//-----------------------------------------------------------------------------

namespace
{

class AddPageUndo : public TUndo
{
	TPaletteHandle *m_paletteHandle;
	TPaletteP m_palette;
	int m_pageIndex;
	wstring m_pageName;
	std::vector<std::pair<TColorStyle *, int>> m_styles;

public:
	// creare DOPO l'inserimento
	AddPageUndo(
		TPaletteHandle *paletteHandle,
		int pageIndex,
		wstring pageName)
		: m_paletteHandle(paletteHandle), m_pageIndex(pageIndex), m_pageName(pageName)
	{
		m_palette = m_paletteHandle->getPalette();
		assert(m_palette);
		assert(0 <= m_pageIndex && m_pageIndex < m_palette->getPageCount());
		TPalette::Page *page = m_palette->getPage(m_pageIndex);
		assert(page);
		for (int i = 0; i < page->getStyleCount(); i++) {
			std::pair<TColorStyle *, int> p;
			p.first = page->getStyle(i)->clone();
			p.second = page->getStyleId(i);
			m_styles.push_back(p);
		}
	}
	~AddPageUndo()
	{
		for (int i = 0; i < (int)m_styles.size(); i++)
			delete m_styles[i].first;
	}
	void undo() const
	{
		m_palette->erasePage(m_pageIndex);
		m_paletteHandle->notifyPaletteChanged();
	}
	void redo() const
	{
		TPalette::Page *page = m_palette->addPage(m_pageName);
		assert(page);
		assert(page->getIndex() == m_pageIndex);
		for (int i = 0; i < (int)m_styles.size(); i++) {
			TColorStyle *cs = m_styles[i].first->clone();
			int styleId = m_styles[i].second;
			assert(m_palette->getStylePage(styleId) == 0);
			m_palette->setStyle(styleId, cs);
			page->addStyle(styleId);
		};
		m_paletteHandle->notifyPaletteChanged();
	}
	int getSize() const { return sizeof *this + m_styles.size() * sizeof(TColorStyle); }

	QString getHistoryString()
	{
		return QObject::tr("Add Page %1 to Palette %2")
			.arg(QString::fromStdWString(m_pageName))
			.arg(QString::fromStdWString(m_palette->getPaletteName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

} // namespace

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

void PaletteCmd::addPage(TPaletteHandle *paletteHandle, wstring name, bool withUndo)
{
	TPalette *palette = paletteHandle->getPalette();
	if (name == L"")
		name = L"page " + toWideString(palette->getPageCount() + 1);
	TPalette::Page *page = palette->addPage(name);

	palette->setDirtyFlag(true);

	paletteHandle->notifyPaletteChanged();
	if (withUndo)
		TUndoManager::manager()->add(new AddPageUndo(paletteHandle, page->getIndex(), name));
}

//=============================================================================
// destroyPage
//-----------------------------------------------------------------------------

namespace
{

class DestroyPageUndo : public TUndo
{
	TPaletteHandle *m_paletteHandle;
	TPaletteP m_palette;
	int m_pageIndex;
	wstring m_pageName;
	std::vector<int> m_styles;

public:
	DestroyPageUndo(TPaletteHandle *paletteHandle, int pageIndex)
		: m_paletteHandle(paletteHandle), m_pageIndex(pageIndex)
	{
		m_palette = m_paletteHandle->getPalette();
		assert(m_palette);
		assert(0 <= pageIndex && pageIndex < m_palette->getPageCount());
		assert(m_palette->getPageCount() > 1);
		TPalette::Page *page = m_palette->getPage(m_pageIndex);
		assert(page);
		m_pageName = page->getName();
		m_styles.resize(page->getStyleCount());
		for (int i = 0; i < page->getStyleCount(); i++)
			m_styles[i] = page->getStyleId(i);
	}
	void undo() const
	{
		TPalette::Page *page = m_palette->addPage(m_pageName);
		m_palette->movePage(page, m_pageIndex);
		for (int i = 0; i < (int)m_styles.size(); i++)
			page->addStyle(m_styles[i]);
		m_paletteHandle->notifyPaletteChanged();
	}
	void redo() const
	{
		m_palette->erasePage(m_pageIndex);
		m_paletteHandle->notifyPaletteChanged();
	}
	int getSize() const { return sizeof *this; }
	QString getHistoryString()
	{
		return QObject::tr("Delete Page %1 from Palette %2")
			.arg(QString::fromStdWString(m_pageName))
			.arg(QString::fromStdWString(m_palette->getPaletteName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

} // namespace

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

void PaletteCmd::destroyPage(TPaletteHandle *paletteHandle, int pageIndex)
{
	TPalette *palette = paletteHandle->getPalette();
	assert(0 <= pageIndex && pageIndex < palette->getPageCount());

	TUndoManager::manager()->add(new DestroyPageUndo(paletteHandle, pageIndex));
	palette->erasePage(pageIndex);

	palette->setDirtyFlag(true);

	paletteHandle->notifyPaletteChanged();
}

//===================================================================
// ReferenceImage
//-------------------------------------------------------------------

namespace
{

//===================================================================
// SetReferenceImageUndo
//-------------------------------------------------------------------

class SetReferenceImageUndo : public TUndo
{
	TPaletteP m_palette;
	TPaletteP m_oldPalette, m_newPalette;

	TPaletteHandle *m_paletteHandle;

public:
	SetReferenceImageUndo(TPaletteP palette, TPaletteHandle *paletteHandle)
		: m_palette(palette), m_oldPalette(palette->clone()), m_paletteHandle(paletteHandle)
	{
	}
	void onAdd()
	{
		m_newPalette = m_palette->clone();
	}
	void undo() const
	{
		m_palette->assign(m_oldPalette.getPointer());
		m_paletteHandle->notifyPaletteChanged();
	}
	void redo() const
	{
		m_palette->assign(m_newPalette.getPointer());
		m_paletteHandle->notifyPaletteChanged();
	}
	int getSize() const
	{
		return sizeof(*this) + sizeof(TPalette) * 2;
	}
	QString getHistoryString()
	{
		return QObject::tr("Load Color Model %1  to Palette %2")
			.arg(QString::fromStdString(m_newPalette->getRefImgPath().getLevelName()))
			.arg(QString::fromStdWString(m_palette->getPaletteName()));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

//===================================================================
// loadRefImage
//-------------------------------------------------------------------

int loadRefImage(TPaletteHandle *paletteHandle, bool replace,
				 TPaletteP levelPalette, const TFilePath &_fp, int &frame, ToonzScene *scene,
				 const std::vector<int> &frames)
{
	bool paletteAlreadyLoaded = false;
	TFilePath fp = scene->decodeFilePath(_fp);
	if (_fp == TFilePath()) {
		paletteAlreadyLoaded = true;
		fp = levelPalette->getRefImgPath();
	}

	TImageP img;
	try {
		TLevelReaderP lr(fp);
		if (fp != TFilePath() && lr) {
			TLevelP level = lr->loadInfo();
			if (level && level->getFrameCount() > 0) {
				TLevel::Iterator it;
				if (!paletteAlreadyLoaded) {
					std::vector<TFrameId> fids;
					for (it = level->begin(); it != level->end(); ++it) {
						if (it->first == -1 || it->first == -2) {
							assert(level->getFrameCount() == 1);
							fids.push_back(0);
							break;
						}
						// if the frame list is empty, store all fids of the level
						if (frames.empty()) {
							fids.push_back(it->first);
							continue;
						}
						// if the frame list is specified, load only the frames matches with the list
						else {
							std::vector<int>::const_iterator framesIt;
							for (framesIt = frames.begin(); framesIt != frames.end(); framesIt++) {
								if (it->first.getNumber() == *framesIt) {
									fids.push_back(it->first);
									break;
								}
							}
						}
					}
					levelPalette->setRefLevelFids(fids);
				}

				if (frame <= 0)
					frame = level->begin()->first.getNumber();

				const TLevel::Table *table = level->getTable();
				TFrameId fid(frame);

				if (table->find(fid) != table->end()) {
					img = lr->getFrameReader(fid)->load();
					if (img && img->getPalette() == 0) {
						if (paletteAlreadyLoaded || level->getPalette() != 0)
							img->setPalette(level->getPalette());
						else if ((fp.getType() == "tzp" || fp.getType() == "tzu"))
							img->setPalette(ToonzImageUtils::loadTzPalette(fp.withType("plt").withNoFrame()));
					}
				}
			}
		} else
			img = levelPalette->getRefImg();
	} catch (TException &e) {
		std::wcout << L"error: " << e.getMessage() << std::endl;
	} catch (...) {
		std::cout << "error for other reasons" << std::endl;
	}

	if (!img)
		return 1;

	if (paletteAlreadyLoaded) {
		img->setPalette(0);
		levelPalette->setRefImgPath(_fp);
		return 0;
	}

	TUndo *undo = new SetReferenceImageUndo(levelPalette, paletteHandle);

	if (!replace) //ret==1)
	{
		TPaletteP imagePalette;
		if (TRasterImageP ri = img) {
			TRaster32P raster = ri->getRaster();
			if (raster) {
				std::set<TPixel32> colors;
				int colorCount = 256;
				if (Preferences::instance()->getPaletteTypeOnLoadRasterImageAsColorModel() == 0)
					/*-- 全ての異なるピクセルの色を別のStyleにする --*/
					TColorUtils::buildPrecisePalette(colors, raster, colorCount);
				else
					/*-- 似ている色をまとめて1つのStyleにする --*/
					TColorUtils::buildPalette(colors, raster, colorCount);
				colors.erase(TPixel::Black); //il nero viene messo dal costruttore della TPalette
				imagePalette = new TPalette();
				std::set<TPixel32>::const_iterator it = colors.begin();
				for (; it != colors.end(); ++it)
					imagePalette->getPage(0)->addStyle(*it);
			}
		} else
			imagePalette = img->getPalette();

		if (imagePalette) {
			std::wstring gName = levelPalette->getGlobalName();
			//Se sto caricando un reference image su una studio palette
			if (!gName.empty()) {
				imagePalette->setGlobalName(gName);
				StudioPalette::instance()->setStylesGlobalNames(imagePalette.getPointer());
			}
			// voglio evitare di sostituire una palette con pochi colori ad una con tanti colori
			/*-- ColorModelの色数が少ないのにOverwriteしようとした場合は、余分の分だけStyleが追加される --*/
			while (imagePalette->getStyleCount() < levelPalette->getStyleCount()) {
				int index = imagePalette->getStyleCount();
				assert(index < levelPalette->getStyleCount());
				TColorStyle *style = levelPalette->getStyle(index)->clone();
				imagePalette->addStyle(style);
			}
			levelPalette->assign(imagePalette.getPointer());
		}
	}

	img->setPalette(0);

	levelPalette->setRefImgPath(_fp);
	if (!replace)
		levelPalette->setDirtyFlag(true);

	TUndoManager::manager()->add(undo);
	paletteHandle->notifyPaletteChanged();

	return 0;
}

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

//===================================================================
// loadReferenceImage
// load frames specified in frames. if frames is empty, then load all frames of the level.
// return values -- 2: failed_to_get_palette, 1: failed_to_get_image, 0: OK
//-------------------------------------------------------------------

int PaletteCmd::loadReferenceImage(TPaletteHandle *paletteHandle, bool replace,
								   const TFilePath &_fp, int &frame, ToonzScene *scene,
								   const std::vector<int> &frames)
{
	TPaletteP levelPalette = paletteHandle->getPalette();
	if (!levelPalette)
		return 2;

	int ret = loadRefImage(paletteHandle, replace, levelPalette, _fp, frame, scene, frames);
	if (ret != 0)
		return ret;

	// when choosing replace(Keep the destination palette), dirty flag is unchanged
	if (!replace)
		levelPalette->setDirtyFlag(true);

	return 0;
}

//===================================================================
// removeReferenceImage
//-------------------------------------------------------------------

void PaletteCmd::removeReferenceImage(TPaletteHandle *paletteHandle)
{
	TPaletteP levelPalette = paletteHandle->getPalette();
	if (!levelPalette)
		return;
	TUndo *undo = new SetReferenceImageUndo(levelPalette, paletteHandle);

	levelPalette->setRefImg(TImageP());
	levelPalette->setRefImgPath(TFilePath());

	std::vector<TFrameId> fids;
	levelPalette->setRefLevelFids(fids);

	levelPalette->setDirtyFlag(true);
	paletteHandle->notifyPaletteChanged();

	TUndoManager::manager()->add(undo);
}

//=============================================================================
// MovePage
//-----------------------------------------------------------------------------

namespace
{

class MovePageUndo : public TUndo
{
	TPaletteHandle *m_paletteHandle;
	TPaletteP m_palette;
	int m_srcIndex;
	int m_dstIndex;

public:
	MovePageUndo(TPaletteHandle *paletteHandle, int srcIndex, int dstIndex)
		: m_paletteHandle(paletteHandle), m_srcIndex(srcIndex), m_dstIndex(dstIndex)
	{
		m_palette = m_paletteHandle->getPalette();
	}
	void undo() const
	{
		m_palette->movePage(m_palette->getPage(m_dstIndex), m_srcIndex);
		m_paletteHandle->notifyPaletteChanged();
	}
	void redo() const
	{
		m_palette->movePage(m_palette->getPage(m_srcIndex), m_dstIndex);
		m_paletteHandle->notifyPaletteChanged();
	}
	int getSize() const { return sizeof *this; }
	QString getHistoryString()
	{
		return QObject::tr("Move Page");
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

} // namespace

void PaletteCmd::movePalettePage(TPaletteHandle *paletteHandle, int srcIndex, int dstIndex)
{
	TPaletteP palette = paletteHandle->getPalette();
	palette->movePage(palette->getPage(srcIndex), dstIndex);
	TUndoManager::manager()->add(new MovePageUndo(paletteHandle, srcIndex, dstIndex));
	paletteHandle->notifyPaletteChanged();
}

//=============================================================================
// RenamePage
//-----------------------------------------------------------------------------

namespace
{

class RenamePageUndo : public TUndo
{
	TPaletteHandle *m_paletteHandle;
	TPaletteP m_palette;
	int m_pageIndex;
	std::wstring m_newName;
	std::wstring m_oldName;

public:
	RenamePageUndo(TPaletteHandle *paletteHandle, int pageIndex, const std::wstring &newName)
		: m_paletteHandle(paletteHandle), m_pageIndex(pageIndex), m_newName(newName)
	{
		m_palette = m_paletteHandle->getPalette();
		m_oldName = m_palette->getPage(pageIndex)->getName();
	}
	void undo() const
	{
		TPalette::Page *page = m_palette->getPage(m_pageIndex);
		assert(page);
		page->setName(m_oldName);
		m_paletteHandle->notifyPaletteChanged();
	}
	void redo() const
	{
		TPalette::Page *page = m_palette->getPage(m_pageIndex);
		assert(page);
		page->setName(m_newName);
		m_paletteHandle->notifyPaletteChanged();
	}
	int getSize() const { return sizeof *this; }
	QString getHistoryString()
	{
		return QObject::tr("Rename Page  %1 > %2")
			.arg(QString::fromStdWString(m_oldName))
			.arg(QString::fromStdWString(m_newName));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

} // namespace

void PaletteCmd::renamePalettePage(TPaletteHandle *paletteHandle, int pageIndex, const std::wstring &newName)
{
	if (!paletteHandle)
		return;
	TPalette *palette = paletteHandle->getPalette();
	if (!palette)
		return;
	if (pageIndex < 0 || pageIndex >= palette->getPageCount())
		return;
	RenamePageUndo *undo = new RenamePageUndo(paletteHandle, pageIndex, newName);
	paletteHandle->notifyPaletteChanged();
	TPalette::Page *page = palette->getPage(pageIndex);
	assert(page);
	page->setName(newName);
	palette->setDirtyFlag(true);
	paletteHandle->notifyPaletteChanged();
	TUndoManager::manager()->add(undo);
}

//=============================================================================
// RenamePaletteStyle
//-----------------------------------------------------------------------------

namespace
{

class RenamePaletteStyleUndo : public TUndo
{
	TPaletteHandle *m_paletteHandle; //Usato nell'undo e nel redo per lanciare la notifica di cambiamento
	int m_styleId;
	TPaletteP m_palette;
	std::wstring m_newName;
	std::wstring m_oldName;

public:
	RenamePaletteStyleUndo(TPaletteHandle *paletteHandle, const std::wstring &newName)
		: m_paletteHandle(paletteHandle), m_newName(newName)
	{
		m_palette = paletteHandle->getPalette();
		assert(m_palette);
		m_styleId = paletteHandle->getStyleIndex();
		TColorStyle *style = m_palette->getStyle(m_styleId);
		assert(style);
		m_oldName = style->getName();
	}
	void undo() const
	{
		TColorStyle *style = m_palette->getStyle(m_styleId);
		assert(style);
		style->setName(m_oldName);
		m_paletteHandle->notifyColorStyleChanged(false);
	}
	void redo() const
	{
		TColorStyle *style = m_palette->getStyle(m_styleId);
		assert(style);
		style->setName(m_newName);
		m_paletteHandle->notifyColorStyleChanged(false);
	}
	int getSize() const { return sizeof *this; }
	QString getHistoryString()
	{
		return QObject::tr("Rename Style#%1 in Palette%2  : %3 > %4")
			.arg(QString::number(m_styleId))
			.arg(QString::fromStdWString(m_palette->getPaletteName()))
			.arg(QString::fromStdWString(m_oldName))
			.arg(QString::fromStdWString(m_newName));
	}
	int getHistoryType()
	{
		return HistoryType::Palette;
	}
};

} // namespace

void PaletteCmd::renamePaletteStyle(TPaletteHandle *paletteHandle, const std::wstring &newName)
{
	if (!paletteHandle)
		return;
	TPalette *palette = paletteHandle->getPalette();
	if (!palette)
		return;
	TColorStyle *style = paletteHandle->getStyle();
	if (!style)
		return;
	if (style->getName() == newName)
		return;

	RenamePaletteStyleUndo *undo = new RenamePaletteStyleUndo(paletteHandle, newName);
	style->setName(newName);
	palette->setDirtyFlag(true);
	paletteHandle->notifyColorStyleChanged(false);
	TUndoManager::manager()->add(undo);
}