Blob Blame Raw


#include "texception.h"
#include "stdfx.h"
#include "tpixelutils.h"
#include "tbasefx.h"
#include "tfxparam.h"
#include "trop.h"

#include "tparamset.h"

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

//    Local namespace stuff

namespace {

enum Status {
  NoPortsConnected = 0,
  Port0Connected   = 1 << 1,
  Port1Connected   = 1 << 2,
  StatusGood       = Port0Connected | Port1Connected
};

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

inline Status operator|(const Status &l, const Status &r) {
  return Status(((int)l | (int)r));
}

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

inline int operator&(const Status &l) { return int(l); }

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

Status getFxStatus(const TRasterFxPort &port0, const TRasterFxPort &port1) {
  Status status                   = NoPortsConnected;
  if (port0.isConnected()) status = status | Port0Connected;
  if (port1.isConnected()) status = status | Port1Connected;
  return status;
}

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

void makeRectCoherent(TRectD &rect, const TPointD &pos) {
  rect -= pos;
  rect.x0 = tfloor(rect.x0);
  rect.y0 = tfloor(rect.y0);
  rect.x1 = tceil(rect.x1);
  rect.y1 = tceil(rect.y1);
  rect += pos;
}
}

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

//========================
//    Glow functions
//------------------------

namespace {
template <typename T>
void fade(TRasterPT<T> ras, double fade, T color)  // Why it is not in TRop..??
{
  if (fade <= 0.0) return;

  double maxChannelValueD = T::maxChannelValue;

  assert(fade <= 1.0);
  ras->lock();
  for (int j = 0; j < ras->getLy(); j++) {
    T *pix    = ras->pixels(j);
    T *endPix = pix + ras->getLx();

    for (; pix < endPix; ++pix) {
      if (pix->m > 0) {
        if (pix->m == T::maxChannelValue) {
          pix->r = troundp(pix->r + fade * (color.r - pix->r));
          pix->g = troundp(pix->g + fade * (color.g - pix->g));
          pix->b = troundp(pix->b + fade * (color.b - pix->b));
          pix->m = troundp(pix->m + fade * (color.m - pix->m));
        } else {
          int val;
          double factor = pix->m / maxChannelValueD;

          val    = troundp(pix->r + fade * (color.r * factor - pix->r));
          pix->r = (val > T::maxChannelValue) ? T::maxChannelValue : val;

          val    = troundp(pix->g + fade * (color.g * factor - pix->g));
          pix->g = (val > T::maxChannelValue) ? T::maxChannelValue : val;

          val    = troundp(pix->b + fade * (color.b * factor - pix->b));
          pix->b = (val > T::maxChannelValue) ? T::maxChannelValue : val;

          val    = troundp(pix->m + fade * (color.m * factor - pix->m));
          pix->m = (val > T::maxChannelValue) ? T::maxChannelValue : val;
        }
      }
    }
  }

  ras->unlock();
}
}

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

//=================
//    Glow Fx
//-----------------

class GlowFx final : public TBaseRasterFx {
  FX_DECLARATION(GlowFx)

  TRasterFxPort m_lighted, m_light;
  TDoubleParamP m_value;
  TDoubleParamP m_brightness;
  TDoubleParamP m_fade;
  TPixelParamP m_color;

public:
  GlowFx()
      : m_value(10.0)
      , m_brightness(100.0)
      , m_color(TPixel::White)
      , m_fade(0.0) {
    m_value->setMeasureName("fxLength");
    m_color->enableMatte(true);
    m_value->setValueRange(0, (std::numeric_limits<double>::max)());
    m_brightness->setValueRange(0, (std::numeric_limits<double>::max)());
    m_fade->setValueRange(0.0, 100.0);
    bindParam(this, "value", m_value);
    bindParam(this, "brightness", m_brightness);
    bindParam(this, "color", m_color);
    bindParam(this, "fade", m_fade);

    addInputPort("Light", m_light);
    addInputPort("Source", m_lighted);
  }

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

  ~GlowFx() {}

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

  bool doGetBBox(double frame, TRectD &bbox,
                 const TRenderSettings &info) override {
    if (getActiveTimeRegion().contains(frame)) {
      if (m_light.isConnected()) {
        TRectD b0, b1;
        bool ret = m_light->doGetBBox(frame, b0, info);
        bbox     = b0.enlarge(tceil(m_value->getValue(frame)));
        if (m_lighted.isConnected()) {
          ret = ret && m_lighted->doGetBBox(frame, b1, info);
          bbox += b1;
        }
        return ret;
      } else if (m_lighted.isConnected()) {
        return m_lighted->doGetBBox(frame, bbox, info);
      }
    }

    return false;
  }

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

  inline void buildLightRects(const TRectD &tileRect, TRectD &inRect,
                              TRectD &outRect, double blur) {
    if (inRect !=
        TConsts::infiniteRectD)  // Could be, if the input light is a zerary Fx
      makeRectCoherent(inRect, tileRect.getP00());

    int blurI = tceil(blur);

    // It seems that the TRop::blur does wrong with these (cuts at the borders).
    // I don't know why - they would be best...
    // TRectD blurOutRect((lightRect).enlarge(blurI) * tileRect);
    // lightRect = ((tileRect).enlarge(blurI) * lightRect);

    // So we revert to the sum of the two
    outRect = inRect = ((tileRect).enlarge(blurI) * inRect) +
                       ((inRect).enlarge(blurI) * tileRect);
  }

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

  void doCompute(TTile &tile, double frame,
                 const TRenderSettings &ri) override {
    Status status = getFxStatus(m_light, m_lighted);

    if (status & NoPortsConnected)
      // If no port, just do nothing :)
      return;

    // Calculate source
    if (status & Port1Connected) m_lighted->compute(tile, frame, ri);

    // Calculate light
    if (status & Port0Connected) {
      // Init light infos
      TDimension tileSize(tile.getRaster()->getSize());
      TRectD tileRect(tile.m_pos, TDimensionD(tileSize.lx, tileSize.ly));

      double scale = sqrt(fabs(ri.m_affine.det()));
      double blur  = m_value->getValue(frame) * scale;

      // Build the light interesting rect
      TRectD lightRect, blurOutRect;
      m_light->getBBox(frame, lightRect, ri);

      buildLightRects(tileRect, lightRect, blurOutRect, blur);

      if ((lightRect.getLx() <= 0) || (lightRect.getLy() <= 0)) return;
      if ((blurOutRect.getLx() <= 0) || (blurOutRect.getLy() <= 0)) return;

      // Calculate the light tile
      TTile lightTile;
      TDimension lightSize(tround(lightRect.getLx()),
                           tround(lightRect.getLy()));
      m_light->allocateAndCompute(lightTile, lightRect.getP00(), lightSize,
                                  tile.getRaster(), frame, ri);

      // Init glow parameters
      TPixel32 color    = m_color->getValue(frame);
      double brightness = m_brightness->getValue(frame) / 100.0;
      double fade       = m_fade->getValue(frame) / 100.0;

      // Now, apply the glow

      // First, deal with the fade
      {
        TRasterP light     = lightTile.getRaster();
        TRaster32P light32 = light;
        TRaster64P light64 = light;
        if (light32)
          ::fade(light32, fade, color);
        else if (light64)
          ::fade(light64, fade, toPixel64(color));
        else
          assert(false);
      }

      // Then, build the blur
      TRasterP blurOut;
      if (blur > 0) {
        // Build a temporary output to the blur
        {
          TRasterP light(lightTile.getRaster());

          blurOut = light->create(tround(blurOutRect.getLx()),
                                  tround(blurOutRect.getLy()));

          // Apply the blur. Please note that SSE2 should not be used for now -
          // I've seen it
          // doing strange things to the blur...
          TPointD displacement(lightRect.getP00() - blurOutRect.getP00());
          TRop::blur(blurOut, light, blur, tround(displacement.x),
                     tround(displacement.y), false);
        }
      } else
        blurOut = lightTile.getRaster();

      // Apply the rgbm scale
      TRop::rgbmScale(blurOut, blurOut, 1, 1, 1, brightness);

      // Apply the add
      {
        TRectD interestingRect(tileRect * blurOutRect);

        TRect interestingTileRect(tround(interestingRect.x0 - tileRect.x0),
                                  tround(interestingRect.y0 - tileRect.y0),
                                  tround(interestingRect.x1 - tileRect.x0) - 1,
                                  tround(interestingRect.y1 - tileRect.y0) - 1);
        TRect interestingBlurRect(
            tround(interestingRect.x0 - blurOutRect.x0),
            tround(interestingRect.y0 - blurOutRect.y0),
            tround(interestingRect.x1 - blurOutRect.x0) - 1,
            tround(interestingRect.y1 - blurOutRect.y0) - 1);

        if ((interestingTileRect.getLx() <= 0) ||
            (interestingTileRect.getLy() <= 0))
          return;
        if ((interestingBlurRect.getLx() <= 0) ||
            (interestingBlurRect.getLy() <= 0))
          return;

        TRasterP tileInterestRas(
            tile.getRaster()->extract(interestingTileRect));
        TRasterP blurInterestRas(blurOut->extract(interestingBlurRect));
        TRop::add(blurInterestRas, tileInterestRas, tileInterestRas);
      }
    }
  }

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

  void doDryCompute(TRectD &rect, double frame,
                    const TRenderSettings &info) override {
    Status status = getFxStatus(m_light, m_lighted);
    if (status & NoPortsConnected) return;

    if (status & Port1Connected) m_lighted->dryCompute(rect, frame, info);

    if (status & Port0Connected) {
      double scale = sqrt(fabs(info.m_affine.det()));
      double blur  = m_value->getValue(frame) * scale;

      TRectD lightRect, blurOutRect;
      m_light->getBBox(frame, lightRect, info);

      buildLightRects(rect, lightRect, blurOutRect, blur);

      if ((lightRect.getLx() <= 0) || (lightRect.getLy() <= 0)) return;
      if ((blurOutRect.getLx() <= 0) || (blurOutRect.getLy() <= 0)) return;

      m_light->dryCompute(lightRect, frame, info);
    }
  }

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

  // Just like the blur
  bool canHandle(const TRenderSettings &info, double frame) override {
    if (m_light.isConnected())
      return (m_value->getValue(frame) == 0) ? true
                                             : isAlmostIsotropic(info.m_affine);

    return true;
  }

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

  int getMemoryRequirement(const TRectD &rect, double frame,
                           const TRenderSettings &info) override {
    double scale = sqrt(fabs(info.m_affine.det()));
    double blur  = m_value->getValue(frame) * scale;

    return TRasterFx::memorySize(rect.enlarge(blur), info.m_bpp);
  }

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

  TFxPort *getXsheetPort() const override { return getInputPort(1); }
};

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

FX_PLUGIN_IDENTIFIER(GlowFx, "glowFx")
//==================================================================