Blob Blame Raw


#include "stdfx.h"
#include "tfxparam.h"
#include "ttzpimagefx.h"
#include "trop.h"
#include "texturefxP.h"

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

class TextureFx : public TStandardRasterFx
{
	FX_PLUGIN_DECLARATION(TextureFx)

	TRasterFxPort m_input;
	TRasterFxPort m_texture;
	TStringParamP m_string;
	TIntEnumParamP m_keep;
	//TIntEnumParamP m_type;
	TIntEnumParamP m_mode;
	TDoubleParamP m_value;

public:
	TextureFx()
		: m_string(L"1,2,3"), m_keep(new TIntEnumParam(0, "Delete"))
		  //, m_type(new TIntEnumParam(0, "Lines & Areas"))
		  ,
		  m_mode(new TIntEnumParam(SUBSTITUTE, "Texture")), m_value(100)
	{
		addInputPort("Source", m_input);
		addInputPort("Texture", m_texture);
		bindParam(this, "indexes", m_string);
		bindParam(this, "keep", m_keep);
		bindParam(this, "mode", m_mode);
		//bindParam(this,"type",  m_type);
		bindParam(this, "value", m_value);
		m_value->setValueRange(0, 100);
		m_keep->addItem(1, "Keep");
		//m_type->addItem(1, "Lines");
		//m_type->addItem(2, "Areas");
		m_mode->addItem(PATTERNTYPE, "Pattern");
		m_mode->addItem(ADD, "Add");
		m_mode->addItem(SUBTRACT, "Subtract");
		m_mode->addItem(MULTIPLY, "Multiply");
		m_mode->addItem(LIGHTEN, "Lighten");
		m_mode->addItem(DARKEN, "Darken");
	}

	~TextureFx(){};

	bool doGetBBox(double frame, TRectD &bBox, const TRenderSettings &info)
	{
		if (m_input.isConnected()) {
			bool ret = m_input->doGetBBox(frame, bBox, info);
			return ret;
		} else {
			bBox = TRectD();
			return false;
		}
	}

	void doCompute(TTile &tile, double frame, const TRenderSettings &ri);
	void doDryCompute(TRectD &rect, double frame, const TRenderSettings &info);

	bool allowUserCacheOnPort(int port) { return port != 0; }

	int getMemoryRequirement(const TRectD &rect, double frame, const TRenderSettings &info);
	bool canHandle(const TRenderSettings &info, double frame) { return true; }
};

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

void TextureFx::doDryCompute(TRectD &rect, double frame, const TRenderSettings &info)
{
	if (!m_input.isConnected())
		return;

	std::vector<std::string> items;
	std::string indexes = ::to_string(m_string->getValue());
	parseIndexes(indexes, items);
	TRenderSettings ri2(info);
	PaletteFilterFxRenderData *PaletteFilterData = new PaletteFilterFxRenderData;
	insertIndexes(items, PaletteFilterData);
	PaletteFilterData->m_keep = (bool)(m_keep->getValue() == 1);

	ri2.m_data.push_back(PaletteFilterData);
	ri2.m_userCachable = false;

	//First child compute: part of output that IS NOT texturized
	m_input->dryCompute(rect, frame, ri2);

	if (!m_texture.isConnected())
		return;

	bool isSwatch = ri2.m_isSwatch;
	if (isSwatch)
		ri2.m_isSwatch = false;
	PaletteFilterData->m_keep = !(m_keep->getValue());

	//Second child compute: part of output that IS to be texturized
	m_input->dryCompute(rect, frame, ri2);

	//Third child compute: texture
	m_texture->dryCompute(rect, frame, info);
}

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

void TextureFx::doCompute(TTile &tile, double frame, const TRenderSettings &ri)
{
	if (!m_input.isConnected())
		return;

	TTile invertMaskTile;

	//carico il vettore items con gli indici dei colori
	std::vector<std::string> items;
	std::string indexes = ::to_string(m_string->getValue());
	parseIndexes(indexes, items);

	//genero il tile il cui raster contiene l'immagine in input a cui sono stati tolti i pixel
	//colorati con gli indici contenuti nel vettore items
	TRenderSettings ri2(ri);
	PaletteFilterFxRenderData *PaletteFilterData = new PaletteFilterFxRenderData;
	PaletteFilterData->m_keep = !!(m_keep->getValue());
	insertIndexes(items, PaletteFilterData);

	ri2.m_data.push_back(PaletteFilterData);
	ri2.m_userCachable = false;

	m_input->allocateAndCompute(invertMaskTile, tile.m_pos, tile.getRaster()->getSize(),
								tile.getRaster(), frame, ri2);

	if (!m_texture.isConnected()) {
		tile.getRaster()->copy(invertMaskTile.getRaster());
		return;
	}

	//genero il tile il cui raster contiene l'immagine in input a cui sono stati tolti i pixel
	//colorati con indici diversi da quelli contenuti nel vettore items
	bool isSwatch = ri2.m_isSwatch;
	if (isSwatch)
		ri2.m_isSwatch = false;
	PaletteFilterData->m_keep = !(m_keep->getValue());
	m_input->compute(tile, frame, ri2);
	if (isSwatch)
		ri2.m_isSwatch = true;

	//controllo se ho ottenuto quaclosa su cui si possa lavorare.
	TRect box;
	TRop::computeBBox(tile.getRaster(), box);
	if (box.isEmpty()) {
		m_input->compute(tile, frame, ri); //Could the invertMask be copied??
		return;
	}

	//Then, generate the texture tile
	TTile textureTile;
	TDimension size = tile.getRaster()->getSize();
	TPointD pos = tile.m_pos;
	m_texture->allocateAndCompute(textureTile, pos, size, tile.getRaster(), frame, ri);

	//And copy the part corresponding to mask tile
	/*TDimension dim = tile.getRaster()->getSize();
  TRasterP appRas = tile.getRaster()->create(dim.lx,dim.ly);
	appRas->clear();
	appRas->copy(textureTile.getRaster(),convert(textureTile.m_pos-tile.m_pos));
	textureTile.setRaster(appRas);*/

	double v = m_value->getValue(frame);
	if (ri.m_bpp == 32) {
		TRaster32P witheCard;

		myOver32(tile.getRaster(), invertMaskTile.getRaster(), &makeOpaque<TPixel32>, v);

		switch (m_mode->getValue()) {
		case SUBSTITUTE:
			myOver32(tile.getRaster(), textureTile.getRaster(), &substitute<TPixel32>, v);
			break;
		case PATTERNTYPE:
			witheCard = TRaster32P(textureTile.getRaster()->getSize());
			witheCard->fill(TPixel32::White);
			TRop::over(textureTile.getRaster(), witheCard, textureTile.getRaster());
			myOver32(tile.getRaster(), textureTile.getRaster(), &pattern32, v);
			break;
		case ADD:
			myOver32(tile.getRaster(), textureTile.getRaster(), &textureAdd<TPixel32>, v / 100.0);
			break;
		case SUBTRACT:
			myOver32(tile.getRaster(), textureTile.getRaster(), &textureSub<TPixel32>, v / 100.0);
			break;
		case MULTIPLY:
			myOver32(tile.getRaster(), textureTile.getRaster(), &textureMult<TPixel32>, v);
			break;
		case DARKEN:
			myOver32(tile.getRaster(), textureTile.getRaster(), &textureDarken<TPixel32>, v);
			break;
		case LIGHTEN : myOver32(tile.getRaster(), textureTile.getRaster(), &textureLighten<TPixel32>, v);
			break;
		default:
			assert(0);
			break;
		}
	} else {
		TRaster64P witheCard;

		myOver64(tile.getRaster(), invertMaskTile.getRaster(), &makeOpaque<TPixel64>, v);

		switch (m_mode->getValue()) {
		case SUBSTITUTE:
			myOver64(tile.getRaster(), textureTile.getRaster(), &substitute<TPixel64>, v);
			break;
		case PATTERNTYPE:
			witheCard = TRaster64P(textureTile.getRaster()->getSize());
			witheCard->fill(TPixel64::White);
			TRop::over(textureTile.getRaster(), witheCard, textureTile.getRaster());
			myOver64(tile.getRaster(), textureTile.getRaster(), &pattern64, v);
			break;
		case ADD:
			myOver64(tile.getRaster(), textureTile.getRaster(), &textureAdd<TPixel64>, v / 100.0);
			break;
		case SUBTRACT:
			myOver64(tile.getRaster(), textureTile.getRaster(), &textureSub<TPixel64>, v / 100.0);
			break;
		case MULTIPLY:
			myOver64(tile.getRaster(), textureTile.getRaster(), &textureMult<TPixel64>, v);
			break;
		case DARKEN:
			myOver64(tile.getRaster(), textureTile.getRaster(), &textureDarken<TPixel64>, v);
			break;
		case LIGHTEN:
			myOver64(tile.getRaster(), textureTile.getRaster(), &textureLighten<TPixel64>, v);
			break;
		default:
			assert(0);
		}
	}

	TRop::over(tile.getRaster(), invertMaskTile.getRaster());
}

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

int TextureFx::getMemoryRequirement(const TRectD &rect, double frame, const TRenderSettings &info)
{
	return TRasterFx::memorySize(rect, info.m_bpp);
}

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

FX_PLUGIN_IDENTIFIER(TextureFx, "textureFx");