Blob Blame Raw


#include "toonz/scriptbinding_image_builder.h"
#include "toonz/scriptbinding_image.h"
#include "ttoonzimage.h"
#include "trop.h"
#include <cmath>

namespace TScriptBinding
{

Transform::Transform()
{
}

Transform::Transform(const TAffine &aff)
	: m_affine(aff)
{
}

Transform::~Transform()
{
}

QScriptValue Transform::ctor(QScriptContext *context, QScriptEngine *engine)
{
	return create(engine, new Transform());
}

QScriptValue Transform::toString()
{
	if (m_affine.isIdentity())
		return tr("Identity");
	else if (m_affine.isTranslation())
		return tr("Translation(%1,%2)").arg(m_affine.a13).arg(m_affine.a23);
	else {
		QString translationPart = "";
		if (m_affine.a13 != 0.0 || m_affine.a23 != 0.0)
			translationPart = tr("Translation(%1,%2)").arg(m_affine.a13).arg(m_affine.a23);

		if (fabs(m_affine.det() - 1.0) < 1.0e-8) {
			double phi = atan2(m_affine.a12, m_affine.a11) * 180.0 / M_PI;
			phi = -(0.001 * floor(1000 * phi + 0.5));
			QString rotationPart = tr("Rotation(%1)").arg(phi);
			if (translationPart == "")
				return rotationPart;
			else
				return translationPart + "*" + rotationPart;
		} else if (m_affine.a12 == 0.0 && m_affine.a21 == 0.0) {
			QString scalePart;
			if (m_affine.a11 == m_affine.a22)
				scalePart = tr("Scale(%1%)").arg(m_affine.a11 * 100);
			else
				scalePart = tr("Scale(%1%, %2%)").arg(m_affine.a11 * 100).arg(m_affine.a22 * 100);
			if (translationPart == "")
				return scalePart;
			else
				return translationPart + "*" + scalePart;
		} else {
			return tr("Transform(%1, %2, %3;  %4, %5, %6)")
				.arg(m_affine.a11)
				.arg(m_affine.a12)
				.arg(m_affine.a13)
				.arg(m_affine.a21)
				.arg(m_affine.a22)
				.arg(m_affine.a23);
		}
	}
}

QScriptValue Transform::translate(double x, double y)
{
	return create(engine(), new Transform(TTranslation(x, y) * m_affine));
}

QScriptValue Transform::rotate(double degrees)
{
	return create(engine(), new Transform(TRotation(degrees) * m_affine));
}

QScriptValue Transform::scale(double s)
{
	return create(engine(), new Transform(TScale(s) * m_affine));
}

QScriptValue Transform::scale(double sx, double sy)
{
	return create(engine(), new Transform(TScale(sx, sy) * m_affine));
}

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

ImageBuilder::ImageBuilder()
	: m_width(0), m_height(0)
{
}

ImageBuilder::~ImageBuilder()
{
}

QScriptValue ImageBuilder::ctor(QScriptContext *context, QScriptEngine *engine)
{
	ImageBuilder *imageBuilder = 0;
	if (context->argumentCount() == 2 || context->argumentCount() == 3) {
		if (!context->argument(0).isNumber() || !context->argument(1).isNumber())
			return context->throwError("Bad arguments: expected width,height[,type]");
		int width = (int)(context->argument(0).toNumber());
		int height = (int)(context->argument(1).toNumber());
		if (width <= 0 || height <= 0)
			return context->throwError("Bad size");
		QString type;
		if (context->argumentCount() == 3) {
			if (context->argument(2).isString())
				type = context->argument(2).toString();
			if (type != "Raster" && type != "ToonzRaster")
				return context->throwError(tr("Bad argument (%1): should be 'Raster' or ToonzRaster'").arg(context->argument(2).toString()));
		}
		imageBuilder = new ImageBuilder();
		imageBuilder->m_width = width;
		imageBuilder->m_height = height;
		if (type == "Raster")
			imageBuilder->m_img = new TRasterImage(TRaster32P(width, height));
		else if (type == "ToonzRaster") {
			imageBuilder->m_img = new TToonzImage(TRasterCM32P(width, height), TRect(0, 0, width, height));
		}
	} else {
		if (context->argumentCount() != 0)
			return context->throwError("Bad argument count. expected: width,height[,type]");
		imageBuilder = new ImageBuilder();
	}
	QScriptValue obj = engine->newQObject(imageBuilder, QScriptEngine::AutoOwnership, QScriptEngine::ExcludeSuperClassContents | QScriptEngine::ExcludeSuperClassMethods);
	return obj;
}

QScriptValue ImageBuilder::toString()
{
	QString type = "Empty";
	if (m_img) {
		if (m_img->getType() == TImage::RASTER)
			type = "Raster";
		else if (m_img->getType() == TImage::TOONZ_RASTER)
			type = "ToonzRaster";
		else if (m_img->getType() == TImage::VECTOR)
			type = "Vector";
		else
			type = "Bad";
	}
	return tr("ImageBuilder(%1 image)").arg(type);
}

QScriptValue ImageBuilder::getImage()
{
	return create(engine(), new Image(m_img));
}

QScriptValue ImageBuilder::fill(const QString &colorName)
{
	QColor color;
	QScriptValue err = checkColor(context(), colorName, color);
	if (err.isError())
		return err;
	TPixel32 pix(color.red(), color.green(), color.blue(), color.alpha());
	if (m_img) {
		if (m_img->getType() != TImage::RASTER)
			context()->throwError("Can't fill a non-'Raster' image");
		TRaster32P ras = m_img->raster();
		if (ras)
			ras->fill(pix);
	} else if (m_width > 0 && m_height > 0) {
		TRaster32P ras(m_width, m_height);
		ras->fill(pix);
		m_img = TRasterImageP(ras);
	}
	return context()->thisObject();
}

void ImageBuilder::clear()
{
	m_img = TImageP();
}

QString ImageBuilder::add(const TImageP &img, const TAffine &aff)
{
	if (fabs(aff.det()) < 0.001)
		return "";
	if (m_img && (m_img->getType() != img->getType()))
		return "Image type mismatch";

	if (!m_img && img->getType() != TImage::VECTOR && m_width > 0 && m_height > 0) {
		TRasterP ras = img->raster()->create(m_width, m_height);
		if (img->getType() == TImage::RASTER)
			m_img = TRasterImageP(ras);
		else if (img->getType() == TImage::TOONZ_RASTER) {
			m_img = TToonzImageP(ras, ras->getBounds());
			m_img->setPalette(img->getPalette());
		} else
			return "Bad image type";
	}

	if (!m_img && aff.isIdentity()) {
		m_img = img->cloneImage();
	} else if (img->getType() == TImage::VECTOR) {
		// vector image
		if (!m_img) {
			// transform
			TVectorImageP vi = img->cloneImage();
			vi->transform(aff);
			m_img = vi;
		} else {
			// merge
			TVectorImageP up = img;
			TVectorImageP dn = m_img;
			dn->mergeImage(up, aff);
		}
	} else {
		if (img->getType() != TImage::TOONZ_RASTER && img->getType() != TImage::RASTER) {
			// this should not ever happen
			return "Bad image type";
		}
		TRasterP up = img->raster();
		if (!m_img) {
			// create an empty bg
			TRasterP ras = up->create();
			ras->clear();
			if (img->getType() == TImage::TOONZ_RASTER) {
				TRasterCM32P rasCm = ras;
				m_img = TToonzImageP(rasCm, TRect(0, 0, ras->getLx() - 1, ras->getLy() - 1));
				m_img->setPalette(img->getPalette());
			} else {
				m_img = TRasterImageP(ras);
			}
		}
		TRasterP dn = m_img->raster();
		if (aff.isTranslation() && aff.a13 == floor(aff.a13) && aff.a23 == floor(aff.a23)) {
			// just a integer coord translation
			TPoint delta = -up->getCenter() + dn->getCenter() + TPoint((int)aff.a13, (int)aff.a23);
			TRop::over(dn, up, delta);
		} else {
			TAffine aff1 = TTranslation(dn->getCenterD()) * aff * TTranslation(-up->getCenterD());
			TRop::over(dn, up, aff1, TRop::Mitchell);
		}
	}
	return "";
}

/*
  TImageP srcImg = img->getImg();
  if(srcImg->getType()==TImage::RASTER || srcImg->getType()==TImage::TOONZ_RASTER)
  {
    TRasterP in = srcImg->raster();
    TRasterP out = in->create();
    TPointD center = in->getCenterD();
    TAffine aff1 = TTranslation(center) * aff * TTranslation(-center);
    TRop::resample(out,in,aff1,TRop::Mitchell);
    m_img = TRasterImageP(out);
  }
  else if(srcImg->getType()==TImage::VECTOR)
  {
    TVectorImageP vi = srcImg->cloneImage();
    vi->transform(aff);
    m_img = vi;
  }
  else
  {
    return context()->throwError(tr("Bad image type"));
  }
  return context()->thisObject();
  */

QScriptValue ImageBuilder::add(QScriptValue imgArg)
{
	Image *simg = 0;
	QScriptValue err = checkImage(context(), imgArg, simg);
	if (err.isError())
		return err;
	QString errStr = add(simg->getImg(), TAffine());
	if (errStr != "")
		return context()->throwError(tr("%1 : %2").arg(errStr).arg(imgArg.toString()));
	else
		return context()->thisObject();
}

QScriptValue ImageBuilder::add(QScriptValue imgArg, QScriptValue transformationArg)
{
	Image *simg = 0;
	QScriptValue err = checkImage(context(), imgArg, simg);
	if (err.isError())
		return err;
	Transform *transformation = qscriptvalue_cast<Transform *>(transformationArg);
	if (!transformation) {
		return context()->throwError(tr("Bad argument (%1): should be a Transformation").arg(transformationArg.toString()));
	}
	TAffine aff = transformation->getAffine();
	QString errStr = add(simg->getImg(), aff);
	if (errStr != "")
		return context()->throwError(tr("%1 : %2").arg(errStr).arg(imgArg.toString()));
	else
		return context()->thisObject();
}

} // namespace TScriptBinding