Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
#include "trasterfx.h"
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
// Core-system includes
Toshihiro Shimizu 890ddd
#include "tsystem.h"
Toshihiro Shimizu 890ddd
#include "tthreadmessage.h"
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
// Fx basics
Toshihiro Shimizu 890ddd
#include "tparamcontainer.h"
Toshihiro Shimizu 890ddd
#include "tbasefx.h"
Toshihiro Shimizu 890ddd
#include "tfxattributes.h"
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
// Images components
Toshihiro Shimizu 890ddd
#include "timagecache.h"
Toshihiro Shimizu 890ddd
#include "trasterimage.h"
Toshihiro Shimizu 890ddd
#include "trop.h"
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
// Optimization components
Toshihiro Shimizu 890ddd
#include "trenderresourcemanager.h"
Toshihiro Shimizu 890ddd
#include "tfxcachemanager.h"
Toshihiro Shimizu 890ddd
#include "trenderer.h"
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
// Diagnostics
Toshihiro Shimizu 890ddd
//#define DIAGNOSTICS
Toshihiro Shimizu 890ddd
#ifdef DIAGNOSTICS
Toshihiro Shimizu 890ddd
#include "diagnostics.h"
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
#endif  // DIAGNOSTICS
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//===================================================
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
namespace {
Shinya Kitaoka 120a6e
inline bool areEqual(const TRasterFxRenderDataP &d1,
Shinya Kitaoka 120a6e
                     const TRasterFxRenderDataP &d2) {
Shinya Kitaoka 120a6e
  return (*d1 == *d2);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
inline int myCeil(double x) {
Shinya Kitaoka 120a6e
  return ((x - (int)(x)) > TConsts::epsilon ? (int)(x) + 1 : (int)(x));
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
inline TRect myConvert(const TRectD &r, TPointD &dp) {
Shinya Kitaoka 120a6e
  TRect ri(tfloor(r.x0), tfloor(r.y0), myCeil(r.x1), myCeil(r.y1));
Shinya Kitaoka 120a6e
  dp.x = r.x0 - ri.x0;
Shinya Kitaoka 120a6e
  dp.y = r.y0 - ri.y0;
Shinya Kitaoka 120a6e
  assert(dp.x >= 0 && dp.y >= 0);
Shinya Kitaoka 120a6e
  return ri;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
inline bool myIsEmpty(const TRectD &r) {
Shinya Kitaoka 120a6e
  return r.getLx() < 1 || r.getLy() < 1;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
inline TRect myConvert(const TRectD &rect) {
Shinya Kitaoka 120a6e
  return TRect(tround(rect.x0), tround(rect.y0), tround(rect.x1) - 1,
Shinya Kitaoka 120a6e
               tround(rect.y1) - 1);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
inline TRectD myConvert(const TRect &rect) {
Shinya Kitaoka 120a6e
  return TRectD(rect.x0, rect.y0, rect.x1 + 1, rect.y1 + 1);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
inline void enlargeToI(TRectD &r) {
Shinya Kitaoka 120a6e
  TRectD temp(tfloor(r.x0), tfloor(r.y0), tceil(r.x1), tceil(r.y1));
Shinya Kitaoka 120a6e
  // NOTE: If we enlarge a TConsts::infiniteRectD or one which trespass
Shinya Kitaoka 120a6e
  // ints' numerical bounds, the rect may become empty.
Shinya Kitaoka 120a6e
  if (!myIsEmpty(temp)) r = temp;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
// Calculates the 2-norm of the passed affine A - that is, the max modulus
Shinya Kitaoka 120a6e
// of A*A's eigenvalues (*A being the adjoint).
Shinya Kitaoka 120a6e
double norm2(const TAffine &aff) {
Shinya Kitaoka 120a6e
  double a11 = aff.a11 * aff.a11 + aff.a12 * aff.a12;
Shinya Kitaoka 120a6e
  double a12 = aff.a11 * aff.a21 + aff.a12 * aff.a22;
Shinya Kitaoka 120a6e
  double a21 = aff.a21 * aff.a11 + aff.a22 * aff.a12;
Shinya Kitaoka 120a6e
  double a22 = aff.a21 * aff.a21 + aff.a22 * aff.a22;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  double a11plusa22 = a11 + a22;
Shinya Kitaoka 120a6e
  double delta      = sq(a11 - a22) + 4 * a12 * a21;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  delta = sqrt(delta);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  double eig1 = a11plusa22 + delta;
Shinya Kitaoka 120a6e
  double eig2 = a11plusa22 - delta;
Shinya Kitaoka 120a6e
  return std::max(sqrt(eig1 / 2.0), sqrt(eig2 / 2.0));
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
inline int getResampleFilterRadius(const TRenderSettings &info) {
Shinya Kitaoka 120a6e
  switch (info.m_quality) {
Shinya Kitaoka 120a6e
  case TRenderSettings::StandardResampleQuality:
Shinya Kitaoka 120a6e
    return 1;
Shinya Kitaoka 120a6e
  case TRenderSettings::ImprovedResampleQuality:
Shinya Kitaoka 120a6e
    return 2;
Shinya Kitaoka 120a6e
  case TRenderSettings::HighResampleQuality:
Shinya Kitaoka 120a6e
    return 3;
Shinya Kitaoka 120a6e
  case TRenderSettings::Triangle_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 1;
Shinya Kitaoka 120a6e
  case TRenderSettings::Mitchell_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 2;
Shinya Kitaoka 120a6e
  case TRenderSettings::Cubic5_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 2;
Shinya Kitaoka 120a6e
  case TRenderSettings::Cubic75_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 2;
Shinya Kitaoka 120a6e
  case TRenderSettings::Cubic1_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 2;
Shinya Kitaoka 120a6e
  case TRenderSettings::Hann2_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 2;
Shinya Kitaoka 120a6e
  case TRenderSettings::Hann3_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 3;
Shinya Kitaoka 120a6e
  case TRenderSettings::Hamming2_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 2;
Shinya Kitaoka 120a6e
  case TRenderSettings::Hamming3_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 3;
Shinya Kitaoka 120a6e
  case TRenderSettings::Lanczos2_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 2;
Shinya Kitaoka 120a6e
  case TRenderSettings::Lanczos3_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 3;
Shinya Kitaoka 120a6e
  case TRenderSettings::Gauss_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 2;
Shinya Kitaoka 120a6e
  case TRenderSettings::ClosestPixel_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 1;
Shinya Kitaoka 120a6e
  case TRenderSettings::Bilinear_FilterResampleQuality:
Shinya Kitaoka 120a6e
    return 1;
Shinya Kitaoka 120a6e
  default:
Shinya Kitaoka 120a6e
    assert(false);
Shinya Kitaoka 120a6e
    return -1;
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
inline QString traduce(const TRectD &rect) {
Shinya Kitaoka 120a6e
  return "[" + QString::number(rect.x0) + " " + QString::number(rect.y0) + " " +
Shinya Kitaoka 120a6e
         QString::number(rect.x1) + " " + QString::number(rect.y1) + "]";
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
inline QString traduce(const TRect &rect) {
Shinya Kitaoka 120a6e
  return "[" + QString::number(rect.x0) + " " + QString::number(rect.y0) + " " +
Shinya Kitaoka 120a6e
         QString::number(rect.x1) + " " + QString::number(rect.y1) + "]";
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
inline QString qTraduce(const TAffine &aff) {
Shinya Kitaoka 120a6e
  return "[" + QString::number(aff.a11, 'g', 15) + "," +
Shinya Kitaoka 120a6e
         QString::number(aff.a12, 'g', 15) + "," +
Shinya Kitaoka 120a6e
         QString::number(aff.a13, 'g', 15) + "," +
Shinya Kitaoka 120a6e
         QString::number(aff.a21, 'g', 15) + "," +
Shinya Kitaoka 120a6e
         QString::number(aff.a22, 'g', 15) + "," +
Shinya Kitaoka 120a6e
         QString::number(aff.a23, 'g', 15) + "]";
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
inline std::string traduce(const TAffine &aff) {
Shinya Kitaoka 120a6e
  return
Shinya Kitaoka 120a6e
      // Observe that toString distinguishes + and - 0. That is a problem
Shinya Kitaoka 120a6e
      // when comparing aliases - so near 0 values are explicitly rounded to 0.
Shinya Kitaoka 120a6e
      (areAlmostEqual(aff.a11, 0.0) ? "0" : ::to_string(aff.a11, 5)) + "," +
Shinya Kitaoka 120a6e
      (areAlmostEqual(aff.a12, 0.0) ? "0" : ::to_string(aff.a12, 5)) + "," +
Shinya Kitaoka 120a6e
      (areAlmostEqual(aff.a13, 0.0) ? "0" : ::to_string(aff.a13, 5)) + "," +
Shinya Kitaoka 120a6e
      (areAlmostEqual(aff.a21, 0.0) ? "0" : ::to_string(aff.a21, 5)) + "," +
Shinya Kitaoka 120a6e
      (areAlmostEqual(aff.a22, 0.0) ? "0" : ::to_string(aff.a22, 5)) + "," +
Shinya Kitaoka 120a6e
      (areAlmostEqual(aff.a23, 0.0) ? "0" : ::to_string(aff.a23, 5));
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
}  // Local namespace
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
// The following should be cleared - and its functionalities surrendered
Shinya Kitaoka 120a6e
// directly
Shinya Kitaoka 120a6e
// to an appropriate resource manager...
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
//! Declares an image to be kept in cache until the render ends or is canceled.
Shinya Kitaoka 120a6e
void addRenderCache(const std::string &alias, TImageP image) {
Shinya Kitaoka 120a6e
  TFxCacheManager::instance()->add(alias, image);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
//! The inverse function to addRenderCache.
Shinya Kitaoka 120a6e
void removeRenderCache(const std::string &alias) {
Shinya Kitaoka 120a6e
  TFxCacheManager::instance()->remove(alias);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//==============================================================================
Toshihiro Shimizu 890ddd
//
Toshihiro Shimizu 890ddd
// TrFx   (Affine Transformer Fx)
Toshihiro Shimizu 890ddd
//
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
//! Internal fx node which applies the implicit affine stored in the
Shinya Kitaoka 120a6e
//! TRenderSettings structure,
Toshihiro Shimizu 890ddd
//! replacing the old TRasterFx::doCompute.
Shinya Kitaoka 120a6e
class TrFx : public TBaseRasterFx {
Shinya Kitaoka 120a6e
  FX_DECLARATION(TrFx)
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  TRasterFx *m_fx;
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
public:
Shinya Kitaoka 120a6e
  TrFx() {}
Shinya Kitaoka 120a6e
  ~TrFx() {}
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  //-----------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  void setFx(TRasterFx *fx) { m_fx = fx; }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  //-----------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 473e70
  bool isCachable() const override {
Shinya Kitaoka 120a6e
    return true;
Shinya Kitaoka 120a6e
  }  // Currently cachable as a test. Observe that it was NOT in Toonz 6.1
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  //-----------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 473e70
  bool canHandle(const TRenderSettings &info, double frame) override { return true; }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  //-----------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 473e70
  std::string getAlias(double frame, const TRenderSettings &info) const override {
Shinya Kitaoka 120a6e
    // NOTE: TrFx are not present at this recursive level. Affines dealing is
Shinya Kitaoka 120a6e
    // currently handled by inserting the
Shinya Kitaoka 120a6e
    // rendering affine AFTER a getAlias call. Ever.
Shinya Kitaoka 120a6e
    std::string alias = getFxType();
Shinya Kitaoka 120a6e
    return alias + "[" + m_fx->getAlias(frame, info) + "]";
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  //-----------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 473e70
  bool doGetBBox(double frame, TRectD &bBox, const TRenderSettings &info) override {
Shinya Kitaoka 120a6e
    // NOTE: TrFx are not present at this recursive level. Affines dealing is
Shinya Kitaoka 120a6e
    // still handled by TGeometryFxs here....
Shinya Kitaoka 120a6e
    return m_fx->doGetBBox(frame, bBox, info);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  //-----------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 473e70
  void doCompute(TTile &tile, double frame, const TRenderSettings &info) override {
Shinya Kitaoka 120a6e
    const TRectD &rectOut =
Shinya Kitaoka 120a6e
        myConvert(tile.getRaster()->getBounds()) + tile.m_pos;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    TRectD rectIn;
Shinya Kitaoka 120a6e
    TRenderSettings infoIn(info);
Shinya Kitaoka 120a6e
    TAffine appliedAff;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    if (!buildInput(rectOut, frame, info, rectIn, infoIn, appliedAff)) return;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    const TRect &rectInI = myConvert(rectIn);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    // rasIn e' un raster dello stesso tipo di tile.getRaster()
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    TTile inTile;
Shinya Kitaoka 120a6e
    m_fx->allocateAndCompute(inTile, rectIn.getP00(),
Shinya Kitaoka 120a6e
                             TDimension(rectInI.getLx(), rectInI.getLy()),
Shinya Kitaoka 120a6e
                             tile.getRaster(), frame, infoIn);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    infoIn.m_affine = appliedAff;
Shinya Kitaoka 120a6e
    TRasterFx::applyAffine(tile, inTile, infoIn);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  //-----------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 473e70
  void doDryCompute(TRectD &rect, double frame, const TRenderSettings &info) override {
Shinya Kitaoka 120a6e
    TRectD rectIn;
Shinya Kitaoka 120a6e
    TRenderSettings infoIn(info);
Shinya Kitaoka 120a6e
    TAffine appliedAff;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    if (!buildInput(rect, frame, info, rectIn, infoIn, appliedAff)) return;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    m_fx->dryCompute(rectIn, frame, infoIn);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  //-----------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  int getMemoryRequirement(const TRectD &rect, double frame,
Shinya Kitaoka 473e70
                           const TRenderSettings &info) override {
Shinya Kitaoka 120a6e
    TRectD rectIn;
Shinya Kitaoka 120a6e
    TRenderSettings infoIn(info);
Shinya Kitaoka 120a6e
    TAffine appliedAff;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    if (!buildInput(rect, frame, info, rectIn, infoIn, appliedAff)) return 0;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    return TRasterFx::memorySize(rectIn, info.m_bpp);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
private:
Shinya Kitaoka 120a6e
  bool buildInput(const TRectD &rectOut, double frame,
Shinya Kitaoka 120a6e
                  const TRenderSettings &infoOut, TRectD &rectIn,
Shinya Kitaoka 120a6e
                  TRenderSettings &infoIn, TAffine &appliedAff) {
Shinya Kitaoka 120a6e
    if (myIsEmpty(rectOut)) return false;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    // Build the affines
Shinya Kitaoka 120a6e
    infoIn.m_affine = m_fx->handledAffine(infoOut, frame);
Shinya Kitaoka 120a6e
    appliedAff      = infoOut.m_affine * infoIn.m_affine.inv();
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    // Quick fix - prevent near-singular transforms.
Shinya Kitaoka 120a6e
    // NOTE: This check **SHOULD BE REMOVED** and dealt with at the RESAMPLING
Shinya Kitaoka 120a6e
    // level.
Shinya Kitaoka 120a6e
    //       Observe how the tolerance value is quite arbitrary! Yeah I know
Shinya Kitaoka 120a6e
    //       it's shameful... :(
Shinya Kitaoka 120a6e
    if (fabs(appliedAff.det()) < 1e-6)  // TO BE REMOVED
Shinya Kitaoka 120a6e
      return false;                     // TO BE REMOVED
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    // Build the input rect
Shinya Kitaoka 120a6e
    const TAffine &appliedAffInv = appliedAff.inv();
Shinya Kitaoka 120a6e
    int filterRadius             = getResampleFilterRadius(infoOut);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    TRectD bbox;
Shinya Kitaoka 120a6e
    m_fx->getBBox(frame, bbox, infoIn);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    rectIn =
Shinya Kitaoka 120a6e
        ((appliedAffInv * rectOut)
Shinya Kitaoka 120a6e
             .enlarge(
Shinya Kitaoka 120a6e
                 filterRadius) +  // The filter size applies in input during
Shinya Kitaoka 120a6e
         (appliedAffInv * rectOut.enlarge(filterRadius))) *
Shinya Kitaoka 120a6e
        bbox;  // magnifications, and in output during
Shinya Kitaoka 120a6e
               // minifications. Thus, they basically cumulate.
Shinya Kitaoka 120a6e
    if (myIsEmpty(rectIn)) return false;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    enlargeToI(rectIn);
Shinya Kitaoka 120a6e
    return true;
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
};
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
FX_IDENTIFIER_IS_HIDDEN(TrFx, "trFx")
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//==============================================================================
Toshihiro Shimizu 890ddd
//
Toshihiro Shimizu 890ddd
// FxResourceBuilder
Toshihiro Shimizu 890ddd
//
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
// This class is the internal interface with the cache regarding intermediate
Shinya Kitaoka 120a6e
// render
Shinya Kitaoka 120a6e
// results. Please refer to the ResourceBuilder documentation in
Shinya Kitaoka 120a6e
// tfxcachemanager.cpp
Shinya Kitaoka 120a6e
class FxResourceBuilder : public ResourceBuilder {
Shinya Kitaoka 120a6e
  TRasterFxP m_rfx;
Shinya Kitaoka 120a6e
  double m_frame;
Shinya Kitaoka 120a6e
  const TRenderSettings *m_rs;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  TTile *m_outTile;
Shinya Kitaoka 120a6e
  TTile *m_currTile;
Shinya Kitaoka 120a6e
  TTile m_newTile;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  TRectD m_outRect;
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
public:
Shinya Kitaoka 120a6e
  FxResourceBuilder(const std::string &resourceName, const TRasterFxP &fx,
Shinya Kitaoka 120a6e
                    const TRenderSettings &rs, double frame)
Shinya Kitaoka 120a6e
      : ResourceBuilder(resourceName, fx.getPointer(), frame, rs)
Shinya Kitaoka 120a6e
      , m_rfx(fx)
Shinya Kitaoka 120a6e
      , m_frame(frame)
Shinya Kitaoka 120a6e
      , m_rs(&rs)
Shinya Kitaoka 120a6e
      , m_currTile(0) {}
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  inline void build(TTile &tile);
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
protected:
Shinya Kitaoka 473e70
  void simCompute(const TRectD &rect) override {
Shinya Kitaoka 120a6e
    TRectD rectCpy(
Shinya Kitaoka 120a6e
        rect);  // Why the hell dryCompute(..) has non-const TRectD& input ????
Shinya Kitaoka 120a6e
    m_rfx->doDryCompute(rectCpy, m_frame, *m_rs);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  void buildTileToCalculate(const TRectD &tileRect);
Shinya Kitaoka 473e70
  void compute(const TRectD &tileRect) override;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 473e70
  void upload(TCacheResourceP &resource) override;
Shinya Kitaoka 473e70
  bool download(TCacheResourceP &resource) override;
Toshihiro Shimizu 890ddd
};
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
inline void FxResourceBuilder::build(TTile &tile) {
Shinya Kitaoka 120a6e
  m_outTile = &tile;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  TDimension dim(tile.getRaster()->getSize());
Shinya Kitaoka 120a6e
  m_outRect = TRectD(tile.m_pos, TDimensionD(dim.lx, dim.ly));
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  ResourceBuilder::build(m_outRect);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void FxResourceBuilder::buildTileToCalculate(const TRectD &tileGeom) {
Shinya Kitaoka 120a6e
  if (tileGeom == m_outRect) {
Shinya Kitaoka 120a6e
    m_currTile = m_outTile;
Shinya Kitaoka 120a6e
    return;
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  m_newTile.m_pos = tileGeom.getP00();
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  TRasterP outRas(m_outTile->getRaster());
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // If possible, try to reuse outRas's buffer.
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  TDimension outputSize(outRas->getSize());
Shinya Kitaoka 120a6e
  TDimension requiredSize(tceil(tileGeom.getLx()), tceil(tileGeom.getLy()));
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  TRasterP ras;
Shinya Kitaoka 120a6e
  if (outputSize.lx >= requiredSize.lx && outputSize.ly >= requiredSize.ly) {
Shinya Kitaoka 120a6e
    // Reuse fxOutput's buffer
Shinya Kitaoka 120a6e
    TRect rect(0, 0, requiredSize.lx - 1, requiredSize.ly - 1);
Shinya Kitaoka 120a6e
    ras = outRas->extract(rect);
Shinya Kitaoka 120a6e
    ras->clear();
Shinya Kitaoka 120a6e
  } else
Shinya Kitaoka 120a6e
    ras = outRas->create(requiredSize.lx, requiredSize.ly);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  m_newTile.setRaster(ras);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  m_currTile = &m_newTile;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void FxResourceBuilder::compute(const TRectD &tileRect) {
Toshihiro Shimizu 890ddd
#ifdef DIAGNOSTICS
Shinya Kitaoka 120a6e
  TStopWatch sw;
Shinya Kitaoka 120a6e
  sw.start();
Toshihiro Shimizu 890ddd
#endif
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  buildTileToCalculate(tileRect);
Shinya Kitaoka 120a6e
  m_rfx->doCompute(*m_currTile, m_frame, *m_rs);
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
#ifdef DIAGNOSTICS
Shinya Kitaoka 120a6e
  sw.stop();
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  DIAGNOSTICS_THRSET("FComputeTime", sw.getTotalTime());
Toshihiro Shimizu 890ddd
#endif
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void FxResourceBuilder::upload(TCacheResourceP &resource) {
Shinya Kitaoka 120a6e
  resource->upload(*m_currTile);
Shinya Kitaoka 120a6e
  if (m_currTile == &m_newTile) m_newTile.setRaster(0);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
bool FxResourceBuilder::download(TCacheResourceP &resource) {
Shinya Kitaoka 120a6e
  // In case the output tile was used to calculate the fx, avoid downloading
Shinya Kitaoka 120a6e
  if (m_currTile == m_outTile) return true;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  return resource->downloadAll(*m_outTile);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//==============================================================================
Toshihiro Shimizu 890ddd
//
Toshihiro Shimizu 890ddd
// TRasterFx
Toshihiro Shimizu 890ddd
//
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
class TRasterFx::TRasterFxImp {
Toshihiro Shimizu 890ddd
public:
Shinya Kitaoka 120a6e
  bool m_cacheEnabled;
Shinya Kitaoka 120a6e
  TTile m_cachedTile;
Shinya Kitaoka 120a6e
  double m_frame;
Shinya Kitaoka 120a6e
  bool m_isEnabled;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  TRenderSettings m_info;
Shinya Kitaoka 120a6e
  std::string m_interactiveCacheId;
Shinya Kitaoka 120a6e
  mutable TThread::Mutex m_mutex;  // brutto
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  TRasterFxImp() : m_cacheEnabled(false), m_isEnabled(true), m_cachedTile(0) {}
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  ~TRasterFxImp() {}
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  void enableCache(bool on) {
Shinya Kitaoka 120a6e
    QMutexLocker sl(&m_mutex);
Shinya Kitaoka 120a6e
    m_cacheEnabled = on;
Shinya Kitaoka 120a6e
    if (!m_cacheEnabled) {
Shinya Kitaoka 120a6e
      m_interactiveCacheId = "";
Shinya Kitaoka 120a6e
      m_frame              = 0;
Shinya Kitaoka 120a6e
      m_info               = TRenderSettings();
Shinya Kitaoka 120a6e
      m_cachedTile.setRaster(0);
Shinya Kitaoka 120a6e
      m_cachedTile.m_pos = TPointD();
Shinya Kitaoka 120a6e
    }
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  bool isCacheEnabled() const { return m_cacheEnabled; }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  bool isEnabled() const {
Shinya Kitaoka 120a6e
    QMutexLocker sl(&m_mutex);  // a che serve??
Shinya Kitaoka 120a6e
    return m_isEnabled;
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  void enable(bool on) {
Shinya Kitaoka 120a6e
    QMutexLocker sl(&m_mutex);  // a che serve
Shinya Kitaoka 120a6e
    m_isEnabled = on;
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
};
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
TRasterFx::TRasterFx() : m_rasFxImp(new TRasterFxImp) {}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
TRasterFx::~TRasterFx() { delete m_rasFxImp; }
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
TAffine TRasterFx::handledAffine(const TRenderSettings &info, double frame) {
Shinya Kitaoka 120a6e
  return (info.m_affine.a11 == info.m_affine.a22 && info.m_affine.a12 == 0 &&
Shinya Kitaoka 120a6e
          info.m_affine.a21 == 0 && info.m_affine.a13 == 0 &&
Shinya Kitaoka 120a6e
          info.m_affine.a23 == 0)
Shinya Kitaoka 120a6e
             ? info.m_affine
Shinya Kitaoka 120a6e
             : TScale(norm2(info.m_affine));
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
bool TRasterFx::getBBox(double frame, TRectD &bBox,
Shinya Kitaoka 120a6e
                        const TRenderSettings &info) {
Shinya Kitaoka 120a6e
  bool ret = doGetBBox(frame, bBox, info);
Shinya Kitaoka 120a6e
  bBox     = info.m_affine * bBox;
Shinya Kitaoka 120a6e
  enlargeToI(bBox);
Shinya Kitaoka 120a6e
  return ret;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void TRasterFx::transform(double frame, int port, const TRectD &rectOnOutput,
Shinya Kitaoka 120a6e
                          const TRenderSettings &infoOnOutput,
Shinya Kitaoka 120a6e
                          TRectD &rectOnInput, TRenderSettings &infoOnInput) {
Shinya Kitaoka 120a6e
  rectOnInput = rectOnOutput;
Shinya Kitaoka 120a6e
  infoOnInput = infoOnOutput;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
int TRasterFx::memorySize(const TRectD &rect, int bpp) {
Shinya Kitaoka 120a6e
  if (rect.x1 <= rect.x0 || rect.y1 <= rect.y0) return 0;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  return (unsigned long)(rect.getLx() + 1) * (unsigned long)(rect.getLy() + 1) *
Shinya Kitaoka 120a6e
             (bpp >> 3) >>
Shinya Kitaoka 120a6e
         20;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
//! Specifies the approximate size (in MegaBytes) of the maximum allocation
Shinya Kitaoka 120a6e
//! instance that this fx
Toshihiro Shimizu 890ddd
//! will need to perform in order to render the passed input.
Shinya Kitaoka 120a6e
//! This method should be reimplemented in order to make the Toonz rendering
Shinya Kitaoka 120a6e
//! process aware of
Toshihiro Shimizu 890ddd
//! the use of big raster memory chunks, at least.
Shinya Kitaoka 120a6e
//! Observe that the passed tile geometry is implicitly  already allocated
Shinya Kitaoka 120a6e
//! <\i> for the fx output.
Toshihiro Shimizu 890ddd
//! \n
Shinya Kitaoka 120a6e
//! The default implementation returns 0, assuming that the passed implicit
Shinya Kitaoka 120a6e
//! memory size is passed
Shinya Kitaoka 120a6e
//! below in node computation without further allocation of resources. Fxs can
Shinya Kitaoka 120a6e
//! reimplement this
Toshihiro Shimizu 890ddd
//! to -1 to explicitly disable the Toonz's smart memory handling.
Toshihiro Shimizu 890ddd
//! \n \n
Shinya Kitaoka 120a6e
//! \note The rendering process ensures that the passed render settings are \a
Shinya Kitaoka 120a6e
//! handled
Toshihiro Shimizu 890ddd
//! by the fx before this function is invoked - do not waste code for it.
Toshihiro Shimizu 890ddd
//! \sa TRasterFx::memorySize and TRasterFx::canHandle methods.
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
int TRasterFx::getMemoryRequirement(const TRectD &rect, double frame,
Shinya Kitaoka 120a6e
                                    const TRenderSettings &info) {
Shinya Kitaoka 120a6e
  return 0;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
std::string TRasterFx::getAlias(double frame,
Shinya Kitaoka 120a6e
                                const TRenderSettings &info) const {
Shinya Kitaoka 120a6e
  std::string alias = getFxType();
Shinya Kitaoka 120a6e
  alias += "[";
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  // alias degli effetti connessi alle porte di input separati da virgole
Shinya Kitaoka 120a6e
  // una porta non connessa da luogo a un alias vuoto (stringa vuota)
Shinya Kitaoka 120a6e
  int i;
Shinya Kitaoka 120a6e
  for (i = 0; i < getInputPortCount(); i++) {
Shinya Kitaoka 120a6e
    TFxPort *port = getInputPort(i);
Shinya Kitaoka 120a6e
    if (port->isConnected()) {
Shinya Kitaoka 120a6e
      TRasterFxP ifx = port->getFx();
Shinya Kitaoka 120a6e
      assert(ifx);
Shinya Kitaoka 120a6e
      alias += ifx->getAlias(frame, info);
Shinya Kitaoka 120a6e
    }
Shinya Kitaoka 120a6e
    alias += ",";
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  // alias dei valori dei parametri dell'effetto al frame dato
Shinya Kitaoka 120a6e
  for (i = 0; i < getParams()->getParamCount(); i++) {
Shinya Kitaoka 120a6e
    TParam *param = getParams()->getParam(i);
Shinya Kitaoka 120a6e
    alias += param->getName() + "=" + param->getValueAlias(frame, 3);
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  alias += "]";
Shinya Kitaoka 120a6e
  return alias;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void TRasterFx::dryCompute(TRectD &rect, double frame,
Shinya Kitaoka 120a6e
                           const TRenderSettings &info) {
Shinya Kitaoka 120a6e
  if (checkActiveTimeRegion() && !getActiveTimeRegion().contains(frame)) return;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  if (!getAttributes()->isEnabled() || !m_rasFxImp->isEnabled()) {
Shinya Kitaoka 120a6e
    if (getInputPortCount() == 0) return;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    TFxPort *port = getInputPort(getPreferredInputPort());
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    if (port->isConnected())
Shinya Kitaoka 120a6e
      TRasterFxP(port->getFx())->dryCompute(rect, frame, info);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    return;
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  // If the input tile has a fractionary position, it is passed to the
Shinya Kitaoka 120a6e
  // rendersettings' accumulated affine.
Shinya Kitaoka 120a6e
  TPoint intTilePos(tfloor(rect.x0), tfloor(rect.y0));
Shinya Kitaoka 120a6e
  TPointD fracTilePos(rect.x0 - intTilePos.x, rect.y0 - intTilePos.y);
Shinya Kitaoka 120a6e
  TPointD fracInfoTranslation(info.m_affine.a13 - fracTilePos.x,
Shinya Kitaoka 120a6e
                              info.m_affine.a23 - fracTilePos.y);
Shinya Kitaoka 120a6e
  TPoint intInfoTranslation(tfloor(fracInfoTranslation.x),
Shinya Kitaoka 120a6e
                            tfloor(fracInfoTranslation.y));
Shinya Kitaoka 120a6e
  TPointD newTilePos(intTilePos.x - intInfoTranslation.x,
Shinya Kitaoka 120a6e
                     intTilePos.y - intInfoTranslation.y);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  if (rect.getP00() != newTilePos) {
Shinya Kitaoka 120a6e
    TRenderSettings newInfo(info);
Shinya Kitaoka 120a6e
    newInfo.m_affine.a13 = fracInfoTranslation.x - intInfoTranslation.x;
Shinya Kitaoka 120a6e
    newInfo.m_affine.a23 = fracInfoTranslation.y - intInfoTranslation.y;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    TRectD newRect(newTilePos, rect.getSize());
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    dryCompute(newRect, frame, newInfo);
Shinya Kitaoka 120a6e
    return;
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  // If the fx can't handle the whole affine passed with the TRenderSettings,
Shinya Kitaoka 120a6e
  // the part
Shinya Kitaoka 120a6e
  // of it that the fx can't handle is retained and applied by an affine
Shinya Kitaoka 120a6e
  // transformer fx (TrFx)
Shinya Kitaoka 120a6e
  // after the node has been computed.
Shinya Kitaoka 120a6e
  bool canHandleAffine =
Shinya Kitaoka 120a6e
      canHandle(info, frame) || (handledAffine(info, frame) == info.m_affine);
Shinya Kitaoka 120a6e
  if (!info.m_affine.isIdentity() && !canHandleAffine) {
Shinya Kitaoka 120a6e
    TrFx *transformerFx = new TrFx;
Shinya Kitaoka 120a6e
    TFxP locker(transformerFx);
Shinya Kitaoka 120a6e
    transformerFx->setFx(this);
Shinya Kitaoka 120a6e
    transformerFx->dryCompute(rect, frame, info);
Shinya Kitaoka 120a6e
    return;
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  std::string alias = getAlias(frame, info) + "[" + ::traduce(info.m_affine) +
Shinya Kitaoka 120a6e
                      "][" + std::to_string(info.m_bpp) + "]";
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  int renderStatus =
Shinya Kitaoka 120a6e
      TRenderer::instance().getRenderStatus(TRenderer::renderId());
Shinya Kitaoka 120a6e
  TFxCacheManager *cacheManager = TFxCacheManager::instance();
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  if (renderStatus == TRenderer::FIRSTRUN) {
Shinya Kitaoka 120a6e
    TRectD bbox;
Shinya Kitaoka 120a6e
    // ret = getBBox... puo' darsi che l'enlarge del trFx (o naturale del bbox)
Shinya Kitaoka 120a6e
    // faccia
Shinya Kitaoka 120a6e
    // diventare TRectD() non vuoto!!
Shinya Kitaoka 120a6e
    getBBox(frame, bbox, info);
Shinya Kitaoka 120a6e
    enlargeToI(bbox);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    TRectD interestingRect(rect * bbox);
Shinya Kitaoka 120a6e
    if (myIsEmpty(interestingRect)) return;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    // Declare the tile to the tiles manager
Shinya Kitaoka 120a6e
    ResourceBuilder::declareResource(alias, this, interestingRect, frame, info);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    doDryCompute(interestingRect, frame, info);
Shinya Kitaoka 120a6e
  } else {
Shinya Kitaoka 120a6e
    TRectD bbox;
Shinya Kitaoka 120a6e
    getBBox(frame, bbox, info);
Shinya Kitaoka 120a6e
    enlargeToI(bbox);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    TRectD interestingRect(rect * bbox);
Shinya Kitaoka 120a6e
    if (myIsEmpty(interestingRect)) return;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    // Invoke the fx-specific simulation process
Shinya Kitaoka 120a6e
    FxResourceBuilder rBuilder(alias, this, info, frame);
Shinya Kitaoka 120a6e
    rBuilder.simBuild(interestingRect);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//--------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
//! Declares the computation scheme of this fx for rendering optimization
Shinya Kitaoka 120a6e
//! purposes.
Shinya Kitaoka 120a6e
//! This is an important function of the Toonz rendering API, and should be
Shinya Kitaoka 120a6e
//! reimplemented
Toshihiro Shimizu 890ddd
//! with the necessary care.
Toshihiro Shimizu 890ddd
//! The Toonz rendering process makes use of this function to enact most of the
Toshihiro Shimizu 890ddd
//! optimization steps about the fx computation, in particular fx caching.
Toshihiro Shimizu 890ddd
//! A correct implementation of this method should follow these rules:
Shinya Kitaoka 120a6e
//! 
  • The invocation of child node computations should be faithfully
  • Shinya Kitaoka 120a6e
    //! reproduced.
    Toshihiro Shimizu 890ddd
    //! 
  • TRasterFx::compute and TRasterFx::allocateAndCompute calls have to be
  • Toshihiro Shimizu 890ddd
    //!      translated to TRasterFx::dryCompute calls.
    Shinya Kitaoka 120a6e
    //! 
  • This fx is intended for precomputation stage, so the hard rendering
  • Shinya Kitaoka 120a6e
    //! code
    Toshihiro Shimizu 890ddd
    //!      should be skipped here.
    Shinya Kitaoka 120a6e
    //! By default, this method raises a dryCompute call to each input port in
    Shinya Kitaoka 120a6e
    //! increasing
    Shinya Kitaoka 120a6e
    //! order, using the TRasterFx::transform method to identify the tiles to be
    Shinya Kitaoka 120a6e
    //! passed
    Toshihiro Shimizu 890ddd
    //! on input precomputation.
    Shinya Kitaoka 120a6e
    void TRasterFx::doDryCompute(TRectD &rect, double frame,
    Shinya Kitaoka 120a6e
                                 const TRenderSettings &info) {
    Shinya Kitaoka 120a6e
      int inputPortCount = getInputPortCount();
    Shinya Kitaoka 120a6e
      for (int i = 0; i < inputPortCount; ++i) {
    Shinya Kitaoka 120a6e
        TFxPort *port = getInputPort(i);
    Shinya Kitaoka 120a6e
        if (port->isConnected()) {
    Shinya Kitaoka 120a6e
          TRectD rectOnInput;
    Shinya Kitaoka 120a6e
          TRenderSettings infoOnInput;
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
          TRasterFxP fx = port->getFx();
    Shinya Kitaoka 120a6e
          transform(frame, i, rect, info, rectOnInput, infoOnInput);
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
          if (!myIsEmpty(rectOnInput))
    Shinya Kitaoka 120a6e
            fx->dryCompute(rectOnInput, frame, infoOnInput);
    Shinya Kitaoka 120a6e
        }
    Shinya Kitaoka 120a6e
      }
    Toshihiro Shimizu 890ddd
    }
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    //--------------------------------------------------
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    //! This is an overloaded member function that deals with
    Toshihiro Shimizu 890ddd
    //! the allocation of an input tile before invoking the TRasterFx::compute
    Toshihiro Shimizu 890ddd
    //! method on it.
    Shinya Kitaoka 120a6e
    void TRasterFx::allocateAndCompute(TTile &tile, const TPointD &pos,
    Shinya Kitaoka 120a6e
                                       const TDimension &size, TRasterP templateRas,
    Shinya Kitaoka 120a6e
                                       double frame, const TRenderSettings &info) {
    Shinya Kitaoka 120a6e
      if (templateRas) {
    Shinya Kitaoka 120a6e
        TRaster32P ras32(templateRas);
    Shinya Kitaoka 120a6e
        TRaster64P ras64(templateRas);
    Shinya Kitaoka 120a6e
        templateRas = 0;  // Release the reference to templateRas before allocation
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
        TRasterP tileRas;
    Shinya Kitaoka 120a6e
        if (ras32)
    Shinya Kitaoka 120a6e
          tileRas = TRaster32P(size.lx, size.ly);
    Shinya Kitaoka 120a6e
        else if (ras64)
    Shinya Kitaoka 120a6e
          tileRas = TRaster64P(size.lx, size.ly);
    Shinya Kitaoka 120a6e
        else {
    Shinya Kitaoka 120a6e
          assert(false);
    Shinya Kitaoka 120a6e
          return;
    Shinya Kitaoka 120a6e
        }
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
        tile.setRaster(tileRas);
    Shinya Kitaoka 120a6e
      } else {
    Shinya Kitaoka 120a6e
        if (info.m_bpp == 32) {
    Shinya Kitaoka 120a6e
          TRaster32P tileRas(size.lx, size.ly);
    Shinya Kitaoka 120a6e
          tile.setRaster(tileRas);
    Shinya Kitaoka 120a6e
        } else if (info.m_bpp == 64) {
    Shinya Kitaoka 120a6e
          TRaster64P tileRas(size.lx, size.ly);
    Shinya Kitaoka 120a6e
          tile.setRaster(tileRas);
    Shinya Kitaoka 120a6e
        } else
    Shinya Kitaoka 120a6e
          assert(false);
    Shinya Kitaoka 120a6e
      }
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      tile.m_pos = pos;
    Shinya Kitaoka 120a6e
      compute(tile, frame, info);
    Toshihiro Shimizu 890ddd
    }
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    //-----------------------------------------------------------------------
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    //! This method supplies the actual fx rendering code.
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    void TRasterFx::compute(TTile &tile, double frame,
    Shinya Kitaoka 120a6e
                            const TRenderSettings &info) {
    Shinya Kitaoka 120a6e
      // If the render was aborted, avoid everything
    Shinya Kitaoka 120a6e
      // if(TRenderer::instance().isAborted(TRenderer::renderId()))
    Shinya Kitaoka 120a6e
      //  throw TException("Render canceled");
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      if (checkActiveTimeRegion() && !getActiveTimeRegion().contains(frame)) return;
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      if (!getAttributes()->isEnabled() || !m_rasFxImp->isEnabled()) {
    Shinya Kitaoka 120a6e
        if (getInputPortCount() == 0) return;
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
        TFxPort *port = getInputPort(0);
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
        // la porta 0 non deve essere una porta di controllo
    Shinya Kitaoka 120a6e
        assert(port->isaControlPort() == false);
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
        if (port->isConnected()) {
    Shinya Kitaoka 120a6e
          TRasterFxP(port->getFx())->compute(tile, frame, info);
    Shinya Kitaoka 120a6e
          return;
    Shinya Kitaoka 120a6e
        }
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
        return;
    Shinya Kitaoka 120a6e
      }
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      // If the input tile has a fractionary position, it is passed to the
    Shinya Kitaoka 120a6e
      // rendersettings' accumulated affine. At the same time, the integer part of
    Shinya Kitaoka 120a6e
      // such affine is transferred to the tile.
    Shinya Kitaoka 120a6e
      TPoint intTilePos(tfloor(tile.m_pos.x), tfloor(tile.m_pos.y));
    Shinya Kitaoka 120a6e
      TPointD fracTilePos(tile.m_pos.x - intTilePos.x, tile.m_pos.y - intTilePos.y);
    Shinya Kitaoka 120a6e
      TPointD fracInfoTranslation(info.m_affine.a13 - fracTilePos.x,
    Shinya Kitaoka 120a6e
                                  info.m_affine.a23 - fracTilePos.y);
    Shinya Kitaoka 120a6e
      TPoint intInfoTranslation(tfloor(fracInfoTranslation.x),
    Shinya Kitaoka 120a6e
                                tfloor(fracInfoTranslation.y));
    Shinya Kitaoka 120a6e
      TPointD newTilePos(intTilePos.x - intInfoTranslation.x,
    Shinya Kitaoka 120a6e
                         intTilePos.y - intInfoTranslation.y);
    Shinya Kitaoka 120a6e
      /*-- 入力タイルの位置が、小数値を持っていた場合 --*/
    Shinya Kitaoka 120a6e
      if (tile.m_pos != newTilePos) {
    Shinya Kitaoka 120a6e
        /*-- RenderSettingsのaffine行列に位置ずれを足しこむ --*/
    Shinya Kitaoka 120a6e
        TRenderSettings newInfo(info);
    Shinya Kitaoka 120a6e
        newInfo.m_affine.a13 = fracInfoTranslation.x - intInfoTranslation.x;
    Shinya Kitaoka 120a6e
        newInfo.m_affine.a23 = fracInfoTranslation.y - intInfoTranslation.y;
    Shinya Kitaoka 120a6e
        /*-- タイルの位置は整数値にする --*/
    Shinya Kitaoka 120a6e
        TPointD oldPos(tile.m_pos);
    Shinya Kitaoka 120a6e
        tile.m_pos = newTilePos;
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
        compute(tile, frame, newInfo);
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
        tile.m_pos = oldPos;
    Shinya Kitaoka 120a6e
        return;
    Shinya Kitaoka 120a6e
      }
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      bool canHandleAffine =
    Shinya Kitaoka 120a6e
          canHandle(info, frame) || (handledAffine(info, frame) == info.m_affine);
    Shinya Kitaoka 120a6e
      if (!info.m_affine.isIdentity() && !canHandleAffine) {
    Shinya Kitaoka 120a6e
        TrFx *transformerFx = new TrFx;
    Shinya Kitaoka 120a6e
        TFxP locker(transformerFx);
    Shinya Kitaoka 120a6e
        transformerFx->setFx(this);
    Shinya Kitaoka 120a6e
        transformerFx->compute(tile, frame, info);
    Shinya Kitaoka 120a6e
        return;
    Shinya Kitaoka 120a6e
      }
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      // Retrieve tile's geometry
    Shinya Kitaoka 120a6e
      TRectD tilePlacement = myConvert(tile.getRaster()->getBounds()) + tile.m_pos;
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      // Build the fx result alias (in other words, its name)
    Shinya Kitaoka 120a6e
      std::string alias = getAlias(frame, info) + "[" + ::traduce(info.m_affine) +
    Shinya Kitaoka 120a6e
                          "][" + std::to_string(info.m_bpp) +
    Shinya Kitaoka 120a6e
                          "]";  // To be moved below
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      TRectD bbox;
    Shinya Kitaoka 120a6e
      getBBox(frame, bbox, info);
    Shinya Kitaoka 120a6e
      enlargeToI(bbox);
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      TRectD interestingRect(tilePlacement * bbox);
    Shinya Kitaoka 120a6e
      if (myIsEmpty(interestingRect)) return;
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      // Extract the interesting tile from requested one
    Shinya Kitaoka 120a6e
      TTile interestingTile;
    Shinya Kitaoka 120a6e
      interestingTile.m_pos = interestingRect.getP00();
    Shinya Kitaoka 120a6e
      TRect interestingRectI(myConvert(interestingRect - tilePlacement.getP00()));
    Shinya Kitaoka 120a6e
      interestingTile.setRaster(tile.getRaster()->extract(interestingRectI));
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    #ifdef DIAGNOSTICS
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
      // 1. Push fx name on call stack
    Shinya Kitaoka 120a6e
      QString fxName = QString::fromStdString(getDeclaration()->getId());
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
      DIAGNOSTICS_PUSH("FName", fxName);
    Shinya Kitaoka 120a6e
      DIAGNOSTICS_THRSET("FComputeTime", 0);
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
      TStopWatch sw;
    Shinya Kitaoka 120a6e
      sw.start();
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    #endif
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
      // Invoke the fx-specific computation process
    Shinya Kitaoka 120a6e
      FxResourceBuilder rBuilder(alias, this, info, frame);
    Shinya Kitaoka 120a6e
      rBuilder.build(interestingTile);
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    #ifdef DIAGNOSTICS
    Shinya Kitaoka 120a6e
      sw.stop();
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
      QString countsStr("#ftimes.txt | 4. " + fxName + " | 4. Calls count");
    Shinya Kitaoka 120a6e
      QString fxStr("fcum_4. " + fxName);
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
      long computeTime      = DIAGNOSTICS_THRGET("FComputeTime");
    Shinya Kitaoka 120a6e
      long fxTime           = sw.getTotalTime();
    Shinya Kitaoka 120a6e
      long fxCumulativeTime = DIAGNOSTICS_GLOGET(fxStr);
    Shinya Kitaoka 120a6e
      long count            = DIAGNOSTICS_GET(countsStr);
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
      // 2. Add this time to fx time, and subtract it from parent time
    Shinya Kitaoka 120a6e
      DIAGNOSTICS_GLOADD(fxStr, computeTime);
    Shinya Kitaoka 120a6e
      DIAGNOSTICS_ADD(countsStr, 1);
    Shinya Kitaoka 120a6e
      DIAGNOSTICS_SET("#ftimes.txt | 4. " + fxName + " | 3. Mean time",
    Shinya Kitaoka 120a6e
                      (fxCumulativeTime + computeTime) / (count + 1));
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
      DIAGNOSTICS_POP("FName", 1);
    Shinya Kitaoka 120a6e
      QString parentFxName = DIAGNOSTICS_STACKGET("FName");
    Shinya Kitaoka 120a6e
      if (!parentFxName.isEmpty())
    Shinya Kitaoka 120a6e
        DIAGNOSTICS_GLOADD("fcum_4. " + parentFxName, -fxTime);
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
      DIAGNOSTICS_GLOADD("fcum_03. Cached Fxs Retrieval", fxTime - computeTime);
    Toshihiro Shimizu 890ddd
    #endif
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    #ifdef WRITEIMAGES
    Shinya Kitaoka 120a6e
      static int iCount = 0;
    Shinya Kitaoka 120a6e
      QString qPath("C:\\butta\\image_" +
    Shinya Kitaoka 120a6e
                    QString::number(++iCount).rightJustified(3, '0') + ".tif");
    Shinya Kitaoka 120a6e
      TImageWriter::save(TFilePath(qPath.toStdWString()), tile.getRaster());
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
      if (iCount >= 35) int aa = 1;
    Toshihiro Shimizu 890ddd
    #endif
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
      /*
    Shinya Kitaoka 120a6e
    return;
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    tryCanceled:
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
    //if(TRenderer::instance().isAborted(TRenderer::renderId()))
    Shinya Kitaoka 120a6e
    //  throw TException("Render canceled");
    Shinya Kitaoka 120a6e
    ;
    Toshihiro Shimizu 890ddd
    */
    Toshihiro Shimizu 890ddd
    }
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    //------------------------------------------------------------------------------
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
    TRasterP TRasterFx::applyAffine(TTile &tileOut, const TTile &tileIn,
    Shinya Kitaoka 120a6e
                                    const TRenderSettings &info) {
    Shinya Kitaoka 120a6e
      TAffine aff = info.m_affine;
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      TRasterP src_ras = tileIn.getRaster();
    Shinya Kitaoka 120a6e
      TRasterP dst_ras = tileOut.getRaster();
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      if (aff.isTranslation()) {
    Shinya Kitaoka 120a6e
        // Check the tile origins' fractionary displacement
    Shinya Kitaoka 120a6e
        TPointD diff(tileOut.m_pos - tileIn.m_pos - TPointD(aff.a13, aff.a23));
    Shinya Kitaoka 120a6e
        double fracX = diff.x - tfloor(diff.x);
    Shinya Kitaoka 120a6e
        double fracY = diff.y - tfloor(diff.y);
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
        if ((fracX < 0.01 || fracX > 0.99) && (fracY < 0.01 || fracY > 0.99)) {
    Shinya Kitaoka 120a6e
          // Just copy part of tileIn into tileOut
    Shinya Kitaoka 120a6e
          TRect geomIn(src_ras->getBounds());
    Shinya Kitaoka 120a6e
          TRect geomOut(dst_ras->getBounds());
    Shinya Kitaoka 120a6e
          TPoint diffI(convert(diff));
    Shinya Kitaoka 120a6e
          geomIn -= diffI;
    Shinya Kitaoka 120a6e
          geomOut += diffI;
    Shinya Kitaoka 120a6e
          geomIn *= dst_ras->getBounds();
    Shinya Kitaoka 120a6e
          geomOut *= src_ras->getBounds();
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
          if (geomIn.isEmpty()) return dst_ras;
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
          TRasterP rasIn(src_ras->extract(geomOut));
    Shinya Kitaoka 120a6e
          TRasterP rasOut(dst_ras->extract(geomIn));
    Shinya Kitaoka 120a6e
          TRop::copy(rasOut, rasIn);
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
          return dst_ras;
    Shinya Kitaoka 120a6e
        }
    Shinya Kitaoka 120a6e
      }
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      TRectD rectIn  = myConvert(src_ras->getBounds()) + tileIn.m_pos;
    Shinya Kitaoka 120a6e
      TRectD rectOut = myConvert(dst_ras->getBounds()) + tileOut.m_pos;
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      TRectD rectInAfter = aff * myConvert(src_ras->getBounds());
    Shinya Kitaoka 120a6e
      TAffine rasterAff  = TTranslation((aff * rectIn).getP00() - rectOut.getP00() -
    Shinya Kitaoka 120a6e
                                       rectInAfter.getP00()) *
    Shinya Kitaoka 120a6e
                          aff;
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      TRop::ResampleFilterType qual;
    Shinya Kitaoka 120a6e
      switch (info.m_quality) {
    Shinya Kitaoka 120a6e
      case TRenderSettings::StandardResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Triangle;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::ImprovedResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Hann2;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::HighResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Hamming3;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Triangle_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Triangle;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Mitchell_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Mitchell;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Cubic5_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Cubic5;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Cubic75_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Cubic75;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Cubic1_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Cubic1;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Hann2_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Hann2;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Hann3_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Hann3;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Hamming2_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Hamming2;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Hamming3_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Hamming3;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Lanczos2_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Lanczos2;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Lanczos3_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Lanczos3;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Gauss_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Gauss;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::ClosestPixel_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::ClosestPixel;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      case TRenderSettings::Bilinear_FilterResampleQuality:
    Shinya Kitaoka 120a6e
        qual = TRop::Bilinear;
    Shinya Kitaoka 120a6e
        break;
    Shinya Kitaoka 120a6e
      default:
    Shinya Kitaoka 120a6e
        assert(false);
    Shinya Kitaoka 120a6e
      }
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      TRop::resample(dst_ras, src_ras, rasterAff, qual);
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      return dst_ras;
    Toshihiro Shimizu 890ddd
    }
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    //--------------------------------------------------
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
    bool TRasterFx::isCacheEnabled() const { return m_rasFxImp->m_cacheEnabled; }
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    //--------------------------------------------------
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
    void TRasterFx::enableCache(bool on) { m_rasFxImp->enableCache(on); }
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    //==============================================================================
    Toshihiro Shimizu 890ddd
    //
    Toshihiro Shimizu 890ddd
    // TRenderSettings
    Toshihiro Shimizu 890ddd
    //
    Toshihiro Shimizu 890ddd
    //------------------------------------------------------------------------------
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    TRenderSettings::TRenderSettings()
    Shinya Kitaoka 120a6e
        : m_gamma(1)
    Shinya Kitaoka 120a6e
        , m_timeStretchFrom(25)
    Shinya Kitaoka 120a6e
        , m_timeStretchTo(25)
    Shinya Kitaoka 120a6e
        , m_stereoscopicShift(0.05)
    Shinya Kitaoka 120a6e
        , m_bpp(32)
    Shinya Kitaoka 120a6e
        , m_maxTileSize((std::numeric_limits<int>::max)())</int>
    Shinya Kitaoka 120a6e
        , m_shrinkX(1)
    Shinya Kitaoka 120a6e
        , m_shrinkY(1)
    Shinya Kitaoka 120a6e
        , m_quality(StandardResampleQuality)
    Shinya Kitaoka 120a6e
        , m_fieldPrevalence(NoField)
    Shinya Kitaoka 120a6e
        , m_stereoscopic(false)
    Shinya Kitaoka 120a6e
        , m_isSwatch(false)
    Shinya Kitaoka 120a6e
        , m_applyShrinkToViewer(false)
    Shinya Kitaoka 120a6e
        , m_userCachable(true)
    Shinya Kitaoka 120a6e
        , m_isCanceled(NULL) {}
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    //------------------------------------------------------------------------------
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
    TRenderSettings::~TRenderSettings() {}
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    //------------------------------------------------------------------------------
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
    std::string TRenderSettings::toString() const {
    Shinya Kitaoka 120a6e
      std::string ss =
    Shinya Kitaoka 120a6e
          std::to_string(m_bpp) + ";" + std::to_string(m_quality) + ";" +
    Shinya Kitaoka 120a6e
          std::to_string(m_gamma) + ";" + std::to_string(m_timeStretchFrom) + ";" +
    Shinya Kitaoka 120a6e
          std::to_string(m_timeStretchTo) + ";" +
    Shinya Kitaoka 120a6e
          std::to_string(m_fieldPrevalence) + ";" + std::to_string(m_shrinkX) +
    Shinya Kitaoka 120a6e
          "," + std::to_string(m_shrinkY) + ";" + std::to_string(m_affine.a11) +
    Shinya Kitaoka 120a6e
          "," + std::to_string(m_affine.a12) + "," + std::to_string(m_affine.a13) +
    Shinya Kitaoka 120a6e
          "," + std::to_string(m_affine.a21) + "," + std::to_string(m_affine.a22) +
    Shinya Kitaoka 120a6e
          "," + std::to_string(m_affine.a23) + ";" + std::to_string(m_maxTileSize) +
    Shinya Kitaoka 120a6e
          ";" + std::to_string(m_isSwatch) + ";" + std::to_string(m_userCachable) +
    Shinya Kitaoka 120a6e
          ";{";
    Shinya Kitaoka 120a6e
      if (!m_data.empty()) {
    Shinya Kitaoka 120a6e
        ss += m_data[0]->toString();
    Shinya Kitaoka 120a6e
        for (int i = 1; i < (int)m_data.size(); i++)
    Shinya Kitaoka 120a6e
          ss += "," + m_data[i]->toString();
    Shinya Kitaoka 120a6e
      }
    Shinya Kitaoka 120a6e
      ss += "}";
    Shinya Kitaoka 120a6e
      return ss;
    Toshihiro Shimizu 890ddd
    }
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    //------------------------------------------------------------------------------
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
    bool TRenderSettings::operator==(const TRenderSettings &rhs) const {
    Shinya Kitaoka 120a6e
      if (m_bpp != rhs.m_bpp || m_quality != rhs.m_quality ||
    Shinya Kitaoka 120a6e
          m_fieldPrevalence != rhs.m_fieldPrevalence ||
    Shinya Kitaoka 120a6e
          m_stereoscopic != rhs.m_stereoscopic ||
    Shinya Kitaoka 120a6e
          m_stereoscopicShift != rhs.m_stereoscopicShift ||
    Shinya Kitaoka 120a6e
          m_gamma != rhs.m_gamma || m_timeStretchFrom != rhs.m_timeStretchFrom ||
    Shinya Kitaoka 120a6e
          m_timeStretchTo != rhs.m_timeStretchTo || m_shrinkX != rhs.m_shrinkX ||
    Shinya Kitaoka 120a6e
          m_shrinkY != rhs.m_shrinkY ||
    Shinya Kitaoka 120a6e
          m_applyShrinkToViewer != rhs.m_applyShrinkToViewer ||
    Shinya Kitaoka 120a6e
          m_maxTileSize != rhs.m_maxTileSize || m_affine != rhs.m_affine ||
    Shinya Kitaoka 120a6e
          m_mark != rhs.m_mark || m_isSwatch != rhs.m_isSwatch ||
    Shinya Kitaoka 120a6e
          m_userCachable != rhs.m_userCachable)
    Shinya Kitaoka 120a6e
        return false;
    Shinya Kitaoka 120a6e
    Shinya Kitaoka 120a6e
      return std::equal(m_data.begin(), m_data.end(), rhs.m_data.begin(), areEqual);
    Toshihiro Shimizu 890ddd
    }
    Toshihiro Shimizu 890ddd
    Toshihiro Shimizu 890ddd
    //------------------------------------------------------------------------------
    Toshihiro Shimizu 890ddd
    Shinya Kitaoka 120a6e
    bool TRenderSettings::operator!=(const TRenderSettings &rhs) const {
    Shinya Kitaoka 120a6e
      return !operator==(rhs);
    Toshihiro Shimizu 890ddd
    }