Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
// TnzCore includes
Toshihiro Shimizu 890ddd
#include "tofflinegl.h"
Toshihiro Shimizu 890ddd
#include "tstroke.h"
Toshihiro Shimizu 890ddd
#include "tpalette.h"
Toshihiro Shimizu 890ddd
#include "tvectorimage.h"
Toshihiro Shimizu 890ddd
#include "tvectorrenderdata.h"
Toshihiro Shimizu 890ddd
#include "tflash.h"
Toshihiro Shimizu 890ddd
#include "texception.h"
Toshihiro Shimizu 890ddd
#include "trasterimage.h"
Toshihiro Shimizu 890ddd
#include "drawutil.h"
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
// TnzBase includes
Toshihiro Shimizu 890ddd
#include "trasterfx.h"
Toshihiro Shimizu 890ddd
#include "tparamuiconcept.h"
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
// TnzLib includes
Toshihiro Shimizu 890ddd
#include "toonz/toonzimageutils.h"
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
// TnzStdfx includes
Toshihiro Shimizu 890ddd
#include "particlesengine.h"
Toshihiro Shimizu 890ddd
#include "particlesmanager.h"
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
#include "particlesfx.h"
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//**************************************************************************
Toshihiro Shimizu 890ddd
//    Particles Fx  implementation
Toshihiro Shimizu 890ddd
//**************************************************************************
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
ParticlesFx::ParticlesFx()
Shinya Kitaoka 120a6e
    : m_source("Texture")
Shinya Kitaoka 120a6e
    , m_control("Control")
Shinya Kitaoka 120a6e
    , source_ctrl_val(0)
Shinya Kitaoka 120a6e
    , bright_thres_val(25)
Shinya Kitaoka 120a6e
    , multi_source_val(false)
Shinya Kitaoka 120a6e
    , center_val(TPointD(0.0, 0.0))
Shinya Kitaoka 120a6e
    , length_val(5.0)
Shinya Kitaoka 120a6e
    , height_val(4.0)
Shinya Kitaoka 120a6e
    , maxnum_val(10.0)
Shinya Kitaoka 120a6e
    , lifetime_val(DoublePair(100., 100.))
Shinya Kitaoka 120a6e
    , lifetime_ctrl_val(0)
Shinya Kitaoka 120a6e
    , column_lifetime_val(false)
Shinya Kitaoka 120a6e
    , startpos_val(1)
Shinya Kitaoka 120a6e
    , randseed_val(1)
Shinya Kitaoka 120a6e
    , gravity_val(0.0)
Shinya Kitaoka 120a6e
    , g_angle_val(0.0)
Shinya Kitaoka 120a6e
    , gravity_ctrl_val(0)
Shinya Kitaoka 120a6e
    //, gravity_radius_val (4)
Shinya Kitaoka 120a6e
    , friction_val(0.0)
Shinya Kitaoka 120a6e
    , friction_ctrl_val(0)
Shinya Kitaoka 120a6e
    , windint_val(0.0)
Shinya Kitaoka 120a6e
    , windangle_val(0.0)
Shinya Kitaoka 120a6e
    , swingmode_val(new TIntEnumParam(SWING_RANDOM, "Random"))
Shinya Kitaoka 120a6e
    , randomx_val(DoublePair(0., 0.))
Shinya Kitaoka 120a6e
    , randomy_val(DoublePair(0., 0.))
Shinya Kitaoka 120a6e
    , randomx_ctrl_val(0)
Shinya Kitaoka 120a6e
    , randomy_ctrl_val(0)
Shinya Kitaoka 120a6e
    , swing_val(DoublePair(0., 0.))
Shinya Kitaoka 120a6e
    , speed_val(DoublePair(0., 10.))
Shinya Kitaoka 120a6e
    , speed_ctrl_val(0)
Shinya Kitaoka 120a6e
    , speeda_val(DoublePair(0., 0.))
Shinya Kitaoka 120a6e
    , speeda_ctrl_val(0)
Shinya Kitaoka 120a6e
    , speeda_use_gradient_val(false)
Shinya Kitaoka 120a6e
    , speedscale_val(false)
Shinya Kitaoka 120a6e
    , toplayer_val(new TIntEnumParam(TOP_YOUNGER, "Younger"))
Shinya Kitaoka 120a6e
    , mass_val(DoublePair(1., 1.))
Shinya Kitaoka 120a6e
    , scale_val(DoublePair(100., 100.))
Shinya Kitaoka 120a6e
    , scale_ctrl_val(0)
Shinya Kitaoka 120a6e
    , scale_ctrl_all_val(false)
Shinya Kitaoka 120a6e
    , rot_val(DoublePair(0., 0.))
Shinya Kitaoka 120a6e
    , rot_ctrl_val(0)
Shinya Kitaoka 120a6e
    , trail_val(DoublePair(0., 0.))
Shinya Kitaoka 120a6e
    , trailstep_val(0.0)
Shinya Kitaoka 120a6e
    , rotswingmode_val(new TIntEnumParam(SWING_RANDOM, "Random"))
Shinya Kitaoka 120a6e
    , rotspeed_val(0.0)
Shinya Kitaoka 120a6e
    , rotsca_val(DoublePair(0., 0.))
Shinya Kitaoka 120a6e
    , rotswing_val(DoublePair(0., 0.))
Shinya Kitaoka 120a6e
    , pathaim_val(false)
Shinya Kitaoka 120a6e
    , opacity_val(DoublePair(0., 100.))
Shinya Kitaoka 120a6e
    , opacity_ctrl_val(0)
Shinya Kitaoka 120a6e
    , trailopacity_val(DoublePair(0., 100.))
Shinya Kitaoka 120a6e
    , scalestep_val(DoublePair(0., 0.))
Shinya Kitaoka 120a6e
    , scalestep_ctrl_val(0)
Shinya Kitaoka 120a6e
    , fadein_val(0.0)
Shinya Kitaoka 120a6e
    , fadeout_val(0.0)
Shinya Kitaoka 120a6e
    , animation_val(new TIntEnumParam(ANIM_HOLD, "Hold Frame"))
Shinya Kitaoka 120a6e
    , step_val(1)
Shinya Kitaoka 120a6e
    , gencol_ctrl_val(0)
Shinya Kitaoka 120a6e
    , gencol_spread_val(0.0)
Shinya Kitaoka 120a6e
    , genfadecol_val(0.0)
Shinya Kitaoka 120a6e
    , fincol_ctrl_val(0)
Shinya Kitaoka 120a6e
    , fincol_spread_val(0.0)
Shinya Kitaoka 120a6e
    , finrangecol_val(0.0)
Shinya Kitaoka 120a6e
    , finfadecol_val(0.0)
Shinya Kitaoka 120a6e
    , foutcol_ctrl_val(0)
Shinya Kitaoka 120a6e
    , foutcol_spread_val(0.0)
Shinya Kitaoka 120a6e
    , foutrangecol_val(0.0)
Shinya Kitaoka 120a6e
    , foutfadecol_val(0.0)
Shinya Kitaoka 120a6e
    , source_gradation_val(false)
Shinya Kitaoka 120a6e
    , pick_color_for_every_frame_val(false)
Shinya Kitaoka 120a6e
    , perspective_distribution_val(false) {
Shinya Kitaoka 120a6e
  addInputPort("Texture1", new TRasterFxPort, 0);
Shinya Kitaoka 120a6e
  addInputPort("Control1", new TRasterFxPort, 1);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  length_val->setMeasureName("fxLength");
Shinya Kitaoka 120a6e
  height_val->setMeasureName("fxLength");
Shinya Kitaoka 120a6e
  center_val->getX()->setMeasureName("fxLength");
Shinya Kitaoka 120a6e
  center_val->getY()->setMeasureName("fxLength");
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  bindParam(this, "source_ctrl", source_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "bright_thres", bright_thres_val);
Shinya Kitaoka 120a6e
  bright_thres_val->setValueRange(0, 255);
Shinya Kitaoka 120a6e
  bindParam(this, "multi_source", multi_source_val);
Shinya Kitaoka 120a6e
  bindParam(this, "center", center_val);
Shinya Kitaoka 120a6e
  bindParam(this, "length", length_val);
Shinya Kitaoka 120a6e
  length_val->setValueRange(1.0, (std::numeric_limits<double>::max)());</double>
Shinya Kitaoka 120a6e
  bindParam(this, "height", height_val);
Shinya Kitaoka 120a6e
  height_val->setValueRange(1.0, (std::numeric_limits<double>::max)());</double>
Shinya Kitaoka 120a6e
  bindParam(this, "birth_rate", maxnum_val);
Shinya Kitaoka 120a6e
  maxnum_val->setValueRange(0.0, (std::numeric_limits<double>::max)());</double>
Shinya Kitaoka 120a6e
  bindParam(this, "lifetime", lifetime_val);
Shinya Kitaoka 120a6e
  lifetime_val->getMin()->setValueRange(0., +3000.);
Shinya Kitaoka 120a6e
  lifetime_val->getMax()->setValueRange(0., +3000.);
Shinya Kitaoka 120a6e
  bindParam(this, "lifetime_ctrl", lifetime_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "column_lifetime", column_lifetime_val);
Shinya Kitaoka 120a6e
  bindParam(this, "starting_frame", startpos_val);
Shinya Kitaoka 120a6e
  bindParam(this, "random_seed", randseed_val);
Shinya Kitaoka 120a6e
  bindParam(this, "gravity", gravity_val);
Shinya Kitaoka 120a6e
  gravity_val->setValueRange(0.0, (std::numeric_limits<double>::max)());</double>
Shinya Kitaoka 120a6e
  bindParam(this, "gravity_angle", g_angle_val);
Shinya Kitaoka 120a6e
  g_angle_val->setMeasureName("angle");
Shinya Kitaoka 120a6e
  bindParam(this, "gravity_ctrl", gravity_ctrl_val);
Shinya Kitaoka 120a6e
  //  bindParam(this,"gravity_radius", gravity_radius_val);
Shinya Kitaoka 120a6e
  //  gravity_radius_val->setValueRange(0,40);
Shinya Kitaoka 120a6e
  bindParam(this, "friction", friction_val);
Shinya Kitaoka 120a6e
  bindParam(this, "friction_ctrl", friction_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "wind", windint_val);
Shinya Kitaoka 120a6e
  bindParam(this, "wind_angle", windangle_val);
Shinya Kitaoka 120a6e
  windangle_val->setMeasureName("angle");
Shinya Kitaoka 120a6e
  bindParam(this, "swing_mode", swingmode_val);
Shinya Kitaoka 120a6e
  swingmode_val->addItem(SWING_SMOOTH, "Smooth");
Shinya Kitaoka 120a6e
  bindParam(this, "scattering_x", randomx_val);
Shinya Kitaoka 120a6e
  randomx_val->getMin()->setMeasureName("fxLength");
Shinya Kitaoka 120a6e
  randomx_val->getMax()->setMeasureName("fxLength");
Shinya Kitaoka 120a6e
  randomx_val->getMin()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  randomx_val->getMax()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  bindParam(this, "scattering_y", randomy_val);
Shinya Kitaoka 120a6e
  randomy_val->getMin()->setMeasureName("fxLength");
Shinya Kitaoka 120a6e
  randomy_val->getMax()->setMeasureName("fxLength");
Shinya Kitaoka 120a6e
  randomy_val->getMin()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  randomy_val->getMax()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  bindParam(this, "scattering_x_ctrl", randomx_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "scattering_y_ctrl", randomy_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "swing", swing_val);
Shinya Kitaoka 120a6e
  swing_val->getMin()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  swing_val->getMax()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  speed_val->getMin()->setMeasureName("fxLength");
Shinya Kitaoka 120a6e
  speed_val->getMax()->setMeasureName("fxLength");
Shinya Kitaoka 120a6e
  bindParam(this, "speed", speed_val);
Shinya Kitaoka 120a6e
  speed_val->getMin()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  speed_val->getMax()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  bindParam(this, "speed_ctrl", speed_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "speed_angle", speeda_val);
Shinya Kitaoka 120a6e
  speeda_val->getMin()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  speeda_val->getMax()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  speeda_val->getMin()->setMeasureName("angle");
Shinya Kitaoka 120a6e
  speeda_val->getMax()->setMeasureName("angle");
Shinya Kitaoka 120a6e
  bindParam(this, "speeda_ctrl", speeda_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "speeda_use_gradient", speeda_use_gradient_val);
Shinya Kitaoka 120a6e
  bindParam(this, "speed_size", speedscale_val);
Shinya Kitaoka 120a6e
  bindParam(this, "top_layer", toplayer_val);
Shinya Kitaoka 120a6e
  toplayer_val->addItem(TOP_OLDER, "Older");
Shinya Kitaoka 120a6e
  toplayer_val->addItem(TOP_SMALLER, "Smaller");
Shinya Kitaoka 120a6e
  toplayer_val->addItem(TOP_BIGGER, "Bigger");
Shinya Kitaoka 120a6e
  toplayer_val->addItem(TOP_RANDOM, "Random");
Shinya Kitaoka 120a6e
  bindParam(this, "mass", mass_val);
Shinya Kitaoka 120a6e
  mass_val->getMin()->setValueRange(0., +1000.);
Shinya Kitaoka 120a6e
  mass_val->getMax()->setValueRange(0., +1000.);
Shinya Kitaoka 120a6e
  bindParam(this, "scale", scale_val);
Shinya Kitaoka 120a6e
  scale_val->getMin()->setValueRange(0., +1000.);
Shinya Kitaoka 120a6e
  scale_val->getMax()->setValueRange(0., +1000.);
Shinya Kitaoka 120a6e
  bindParam(this, "scale_ctrl", scale_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "scale_ctrl_all", scale_ctrl_all_val);
Shinya Kitaoka 120a6e
  bindParam(this, "rot", rot_val);
Shinya Kitaoka 120a6e
  rot_val->getMin()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  rot_val->getMax()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  rot_val->getMin()->setMeasureName("angle");
Shinya Kitaoka 120a6e
  rot_val->getMax()->setMeasureName("angle");
Shinya Kitaoka 120a6e
  bindParam(this, "rot_ctrl", rot_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "trail", trail_val);
Shinya Kitaoka 120a6e
  trail_val->getMin()->setValueRange(0., +1000.);
Shinya Kitaoka 120a6e
  trail_val->getMax()->setValueRange(0., +1000.);
Shinya Kitaoka 120a6e
  bindParam(this, "trail_step", trailstep_val);
Shinya Kitaoka 120a6e
  trailstep_val->setValueRange(1.0, (std::numeric_limits<double>::max)());</double>
Shinya Kitaoka 120a6e
  bindParam(this, "spin_swing_mode", rotswingmode_val);
Shinya Kitaoka 120a6e
  rotswingmode_val->addItem(SWING_SMOOTH, "Smooth");
Shinya Kitaoka 120a6e
  bindParam(this, "spin_speed", rotspeed_val);
Shinya Kitaoka 120a6e
  rotspeed_val->setMeasureName("angle");
Shinya Kitaoka 120a6e
  bindParam(this, "spin_random", rotsca_val);
Shinya Kitaoka 120a6e
  rotsca_val->getMin()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  rotsca_val->getMax()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  rotsca_val->getMin()->setMeasureName("angle");
Shinya Kitaoka 120a6e
  rotsca_val->getMax()->setMeasureName("angle");
Shinya Kitaoka 120a6e
  bindParam(this, "spin_swing", rotswing_val);
Shinya Kitaoka 120a6e
  rotswing_val->getMin()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  rotswing_val->getMax()->setValueRange(-1000., +1000.);
Shinya Kitaoka 120a6e
  rotswing_val->getMin()->setMeasureName("angle");
Shinya Kitaoka 120a6e
  rotswing_val->getMax()->setMeasureName("angle");
Shinya Kitaoka 120a6e
  bindParam(this, "path_aim", pathaim_val);
Shinya Kitaoka 120a6e
  bindParam(this, "opacity", opacity_val);
Shinya Kitaoka 120a6e
  opacity_val->getMin()->setValueRange(0., +100.);
Shinya Kitaoka 120a6e
  opacity_val->getMax()->setValueRange(0., +100.);
Shinya Kitaoka 120a6e
  bindParam(this, "opacity_ctrl", opacity_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "trail_opacity", trailopacity_val);
Shinya Kitaoka 120a6e
  trailopacity_val->getMin()->setValueRange(0., +100.);
Shinya Kitaoka 120a6e
  trailopacity_val->getMax()->setValueRange(0., +100.);
Shinya Kitaoka 120a6e
  bindParam(this, "scale_step", scalestep_val);
Shinya Kitaoka 120a6e
  bindParam(this, "scale_step_ctrl", scalestep_ctrl_val);
Shinya Kitaoka 120a6e
  scalestep_val->getMin()->setValueRange(-100., +100.);
Shinya Kitaoka 120a6e
  scalestep_val->getMax()->setValueRange(-100., +100.);
Shinya Kitaoka 120a6e
  bindParam(this, "fade_in", fadein_val);
Shinya Kitaoka 120a6e
  bindParam(this, "fade_out", fadeout_val);
Shinya Kitaoka 120a6e
  bindParam(this, "animation", animation_val);
Shinya Kitaoka 120a6e
  animation_val->addItem(ANIM_RANDOM, "Random Frame");
Shinya Kitaoka 120a6e
  animation_val->addItem(ANIM_CYCLE, "Column");
Shinya Kitaoka 120a6e
  animation_val->addItem(ANIM_R_CYCLE, "Column - Random Start");
Shinya Kitaoka 120a6e
  animation_val->addItem(ANIM_SR_CYCLE, "Column Swing - Random Start");
Shinya Kitaoka 120a6e
  bindParam(this, "step", step_val);
Shinya Kitaoka 120a6e
  step_val->setValueRange(1, (std::numeric_limits<int>::max)());</int>
Shinya Kitaoka 120a6e
  TSpectrum::ColorKey colors[] = {TSpectrum::ColorKey(0, TPixel32::Red),
Shinya Kitaoka 120a6e
                                  TSpectrum::ColorKey(1, TPixel32::Red)};
Shinya Kitaoka 120a6e
  gencol_val = TSpectrumParamP(tArrayCount(colors), colors);
Shinya Kitaoka 120a6e
  bindParam(this, "birth_color", gencol_val);
Shinya Kitaoka 120a6e
  bindParam(this, "birth_color_ctrl", gencol_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "birth_color_spread", gencol_spread_val);
Shinya Kitaoka 120a6e
  gencol_spread_val->setValueRange(0.0, (std::numeric_limits<int>::max)());</int>
Shinya Kitaoka 120a6e
  bindParam(this, "birth_color_fade", genfadecol_val);
Shinya Kitaoka 120a6e
  genfadecol_val->setValueRange(0.0, 100.0);
Shinya Kitaoka 120a6e
  TSpectrum::ColorKey colors1[] = {TSpectrum::ColorKey(0, TPixel32::Green),
Shinya Kitaoka 120a6e
                                   TSpectrum::ColorKey(1, TPixel32::Green)};
Shinya Kitaoka 120a6e
  fincol_val = TSpectrumParamP(tArrayCount(colors1), colors1);
Shinya Kitaoka 120a6e
  bindParam(this, "fadein_color", fincol_val);
Shinya Kitaoka 120a6e
  bindParam(this, "fadein_color_ctrl", fincol_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "fadein_color_spread", fincol_spread_val);
Shinya Kitaoka 120a6e
  fincol_spread_val->setValueRange(0.0, (std::numeric_limits<int>::max)());</int>
Shinya Kitaoka 120a6e
  bindParam(this, "fadein_color_range", finrangecol_val);
Shinya Kitaoka 120a6e
  finrangecol_val->setValueRange(0.0, (std::numeric_limits<double>::max)());</double>
Shinya Kitaoka 120a6e
  bindParam(this, "fadein_color_fade", finfadecol_val);
Shinya Kitaoka 120a6e
  finfadecol_val->setValueRange(0.0, 100.0);
Shinya Kitaoka 120a6e
  TSpectrum::ColorKey colors2[] = {TSpectrum::ColorKey(0, TPixel32::Blue),
Shinya Kitaoka 120a6e
                                   TSpectrum::ColorKey(1, TPixel32::Blue)};
Shinya Kitaoka 120a6e
  foutcol_val = TSpectrumParamP(tArrayCount(colors2), colors2);
Shinya Kitaoka 120a6e
  bindParam(this, "fadeout_color", foutcol_val);
Shinya Kitaoka 120a6e
  bindParam(this, "fadeout_color_ctrl", foutcol_ctrl_val);
Shinya Kitaoka 120a6e
  bindParam(this, "fadeout_color_spread", foutcol_spread_val);
Shinya Kitaoka 120a6e
  foutcol_spread_val->setValueRange(0.0, (std::numeric_limits<int>::max)());</int>
Shinya Kitaoka 120a6e
  bindParam(this, "fadeout_color_range", foutrangecol_val);
Shinya Kitaoka 120a6e
  foutrangecol_val->setValueRange(0.0, (std::numeric_limits<double>::max)());</double>
Shinya Kitaoka 120a6e
  bindParam(this, "fadeout_color_fade", foutfadecol_val);
Shinya Kitaoka 120a6e
  foutfadecol_val->setValueRange(0.0, 100.0);
Shinya Kitaoka 120a6e
  bindParam(this, "source_gradation", source_gradation_val);
Shinya Kitaoka 120a6e
  bindParam(this, "pick_color_for_every_frame", pick_color_for_every_frame_val);
Shinya Kitaoka 120a6e
  bindParam(this, "perspective_distribution", perspective_distribution_val);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
ParticlesFx::~ParticlesFx() {}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void ParticlesFx::getParamUIs(TParamUIConcept *&concepts, int &length) {
Shinya Kitaoka 120a6e
  concepts = new TParamUIConcept[length = 2];
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  concepts[0].m_type  = TParamUIConcept::POINT;
Shinya Kitaoka 120a6e
  concepts[0].m_label = "Center";
Shinya Kitaoka 120a6e
  concepts[0].m_params.push_back(center_val);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  concepts[1].m_type = TParamUIConcept::RECT;
Shinya Kitaoka 120a6e
  concepts[1].m_params.push_back(length_val);
Shinya Kitaoka 120a6e
  concepts[1].m_params.push_back(height_val);
Shinya Kitaoka 120a6e
  concepts[1].m_params.push_back(center_val);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
bool ParticlesFx::doGetBBox(double frame, TRectD &bBox,
Shinya Kitaoka 120a6e
                            const TRenderSettings &info) {
Shinya Kitaoka 120a6e
  // Returning an infinite rect. This is necessary since building the actual
Shinya Kitaoka 120a6e
  // bbox
Shinya Kitaoka 120a6e
  // is a very complicate task.
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  bBox = TConsts::infiniteRectD;
Shinya Kitaoka 120a6e
  return true;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
std::string ParticlesFx::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
  for (int 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
  std::string paramalias("");
Shinya Kitaoka 120a6e
  for (int i = 0; i < getParams()->getParamCount(); ++i) {
Shinya Kitaoka 120a6e
    TParam *param = getParams()->getParam(i);
Shinya Kitaoka 120a6e
    paramalias += param->getName() + "=" + param->getValueAlias(frame, 3);
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  return alias + std::to_string(frame) + "," + std::to_string(getIdentifier()) +
Shinya Kitaoka 120a6e
         paramalias + "]";
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
bool ParticlesFx::allowUserCacheOnPort(int portNum) {
Shinya Kitaoka 120a6e
  // Only control port are currently allowed to cache upon explicit user's
Shinya Kitaoka 120a6e
  // request
Shinya Kitaoka 120a6e
  std::string tmpName = getInputPortName(portNum);
Shinya Kitaoka 120a6e
  return tmpName.find("Control") != std::string::npos;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void ParticlesFx::doDryCompute(TRectD &rect, double frame,
Shinya Kitaoka 120a6e
                               const TRenderSettings &info) {
Shinya Kitaoka 120a6e
  ParticlesManager *pc = ParticlesManager::instance();
Shinya Kitaoka 120a6e
  unsigned long fxId   = getIdentifier();
Shinya Kitaoka 120a6e
  int inputPortCount   = getInputPortCount();
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  int i, j, curr_frame = frame, startframe = startpos_val->getValue();
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  TRenderSettings infoOnInput(info);
Shinya Kitaoka 120a6e
  infoOnInput.m_affine =
Shinya Kitaoka 120a6e
      TAffine();  // Using the standard reference - indep. from cameras.
Shinya Kitaoka 120a6e
  infoOnInput.m_bpp =
Shinya Kitaoka 120a6e
      32;  // Control ports rendered at 32 bit - since not visible.
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  for (i = startframe - 1; i <= curr_frame; ++i) {
Shinya Kitaoka 120a6e
    double frame = std::max(0, i);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    for (j = 0; j < inputPortCount; ++j) {
Shinya Kitaoka 120a6e
      TFxPort *port       = getInputPort(j);
Shinya Kitaoka 120a6e
      std::string tmpName = getInputPortName(j);
Shinya Kitaoka 120a6e
      if (port->isConnected()) {
Shinya Kitaoka 120a6e
        TRasterFxP fx = port->getFx();
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
        // Now, consider that source ports work different than control ones
Shinya Kitaoka 120a6e
        QString portName = QString::fromStdString(tmpName);
Shinya Kitaoka 120a6e
        if (portName.startsWith("C")) {
Shinya Kitaoka 120a6e
          // Control ports are calculated from start to current frame, since
Shinya Kitaoka 120a6e
          // particle mechanics at current frame is influenced by previous ones
Shinya Kitaoka 120a6e
          // (and therefore by all previous control images).
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
          TRectD bbox;
Shinya Kitaoka 120a6e
          fx->getBBox(frame, bbox, infoOnInput);
Shinya Kitaoka 120a6e
          if (bbox == TConsts::infiniteRectD) bbox = info.m_affine.inv() * rect;
Shinya Kitaoka 120a6e
          fx->dryCompute(bbox, frame, infoOnInput);
Shinya Kitaoka 120a6e
        } else if (portName.startsWith("T")) {
Shinya Kitaoka 120a6e
          // Particles handle source ports caching procedures on its own.
Shinya Kitaoka 120a6e
        }
Shinya Kitaoka 120a6e
      }
Shinya Kitaoka 120a6e
    }
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void ParticlesFx::doCompute(TTile &tile, double frame,
Shinya Kitaoka 120a6e
                            const TRenderSettings &ri) {
Shinya Kitaoka 120a6e
  std::vector<int> lastframe;</int>
Shinya Kitaoka 120a6e
  std::vector<tlevelp> partLevel;</tlevelp>
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  TPointD p_offset;
Shinya Kitaoka 120a6e
  TDimension p_size(0, 0);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  /*-- 参照画像ポートの取得 --*/
Shinya Kitaoka 120a6e
  std::vector<trasterfxport *=""> part_ports; /*- テクスチャ素材画像のポート -*/</trasterfxport>
Shinya Kitaoka 120a6e
  std::map<int, *="" trasterfxport=""></int,>
Shinya Kitaoka 120a6e
      ctrl_ports; /*- コントロール画像のポート番号/ポート -*/
Shinya Kitaoka 120a6e
  int portsCount = this->getInputPortCount();
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  for (int i = 0; i < portsCount; ++i) {
Shinya Kitaoka 120a6e
    std::string tmpName = this->getInputPortName(i);
Shinya Kitaoka 120a6e
    QString portName    = QString::fromStdString(tmpName);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    if (portName.startsWith("T")) {
Shinya Kitaoka 120a6e
      TRasterFxPort *tmpPart = (TRasterFxPort *)this->getInputPort(tmpName);
Shinya Kitaoka 120a6e
      if (tmpPart->isConnected())
Shinya Kitaoka 120a6e
        part_ports.push_back((TRasterFxPort *)this->getInputPort(tmpName));
Shinya Kitaoka 120a6e
    } else {
Shinya Kitaoka 120a6e
      portName.replace(QString("Control"), QString(""));
Shinya Kitaoka 120a6e
      TRasterFxPort *tmpCtrl = (TRasterFxPort *)this->getInputPort(tmpName);
Shinya Kitaoka 120a6e
      if (tmpCtrl->isConnected())
Shinya Kitaoka 120a6e
        ctrl_ports[portName.toInt()] =
Shinya Kitaoka 120a6e
            (TRasterFxPort *)this->getInputPort(tmpName);
Shinya Kitaoka 120a6e
    }
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
  /*-- テクスチャ素材のバウンディングボックスを足し合わせる --*/
Shinya Kitaoka 120a6e
  if (!part_ports.empty()) {
Shinya Kitaoka 120a6e
    TRectD outTileBBox(tile.m_pos, TDimensionD(tile.getRaster()->getLx(),
Shinya Kitaoka 120a6e
                                               tile.getRaster()->getLy()));
Shinya Kitaoka 120a6e
    TRectD bbox;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    for (unsigned int i = 0; i < (int)part_ports.size(); ++i) {
Shinya Kitaoka 120a6e
      const TFxTimeRegion &tr = (*part_ports[i])->getTimeRegion();
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
      lastframe.push_back(tr.getLastFrame() + 1);
Shinya Kitaoka 120a6e
      partLevel.push_back(new TLevel());
Shinya Kitaoka 120a6e
      partLevel[i]->setName((*part_ports[i])->getAlias(0, ri));
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
      // The particles offset must be calculated without considering the
Shinya Kitaoka 120a6e
      // affine's translational
Shinya Kitaoka 120a6e
      // component
Shinya Kitaoka 120a6e
      TRenderSettings riZero(ri);
Shinya Kitaoka 120a6e
      riZero.m_affine.a13 = riZero.m_affine.a23 = 0;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
      // Calculate the bboxes union
Shinya Kitaoka 120a6e
      for (int t = 0; t <= tr.getLastFrame(); ++t) {
Shinya Kitaoka 120a6e
        TRectD inputBox;
Shinya Kitaoka 120a6e
        (*part_ports[i])->getBBox(t, inputBox, riZero);
Shinya Kitaoka 120a6e
        bbox += inputBox;
Shinya Kitaoka 120a6e
      }
Shinya Kitaoka 120a6e
    }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    if (bbox == TConsts::infiniteRectD) bbox *= outTileBBox;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    p_size.lx = (int)bbox.getLx() + 1;
Shinya Kitaoka 120a6e
    p_size.ly = (int)bbox.getLy() + 1;
Shinya Kitaoka 120a6e
    p_offset  = TPointD(0.5 * (bbox.x0 + bbox.x1), 0.5 * (bbox.y0 + bbox.y1));
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
  /*- テクスチャ素材が無い場合、丸を描く -*/
Shinya Kitaoka 120a6e
  else {
Shinya Kitaoka 120a6e
    partLevel.push_back(new TLevel());
Shinya Kitaoka 120a6e
    partLevel[0]->setName("particles");
Shinya Kitaoka 120a6e
    TDimension vecsize(10, 10);
Shinya Kitaoka 120a6e
    TOfflineGL *offlineGlContext = new TOfflineGL(vecsize);
Shinya Kitaoka 120a6e
    offlineGlContext->clear(TPixel32(0, 0, 0, 0));
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    TStroke *stroke;
Shinya Kitaoka 120a6e
    stroke = makeEllipticStroke(
Shinya Kitaoka 120a6e
        0.07, TPointD((vecsize.lx - 1) * .5, (vecsize.ly - 1) * .5), 2.0, 2.0);
Shinya Kitaoka 120a6e
    TVectorImageP vectmp = new TVectorImage();
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    TPalette *plt = new TPalette();
Shinya Kitaoka 120a6e
    vectmp->setPalette(plt);
Shinya Kitaoka 120a6e
    vectmp->addStroke(stroke);
Shinya Kitaoka 120a6e
    TVectorRenderData rd(AffI, TRect(vecsize), plt, 0, true, true);
Shinya Kitaoka 120a6e
    offlineGlContext->makeCurrent();
Shinya Kitaoka 120a6e
    offlineGlContext->draw(vectmp, rd);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    partLevel[0]->setFrame(
Shinya Kitaoka 120a6e
        0, TRasterImageP(offlineGlContext->getRaster()->clone()));
Shinya Kitaoka 120a6e
    p_size.lx = vecsize.lx + 1;
Shinya Kitaoka 120a6e
    p_size.ly = vecsize.ly + 1;
Shinya Kitaoka 120a6e
    lastframe.push_back(1);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    delete offlineGlContext;
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  Particles_Engine myEngine(this, frame);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  // Retrieving the dpi multiplier from the accumulated affine (which is
Shinya Kitaoka 120a6e
  // isotropic). That is,
Shinya Kitaoka 120a6e
  // the affine will be applied *before* this effect - and we'll multiply
Shinya Kitaoka 120a6e
  // geometrical parameters
Shinya Kitaoka 120a6e
  // by this dpi mult. in order to compensate.
Shinya Kitaoka 120a6e
  float dpi = sqrt(fabs(ri.m_affine.det())) * 100;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  TTile tileIn;
Shinya Kitaoka 120a6e
  if (TRaster32P raster32 = tile.getRaster()) {
Shinya Kitaoka 120a6e
    TFlash *flash = 0;
Shinya Kitaoka 120a6e
    myEngine.render_particles(flash, &tile, part_ports, ri, p_size, p_offset,
Shinya Kitaoka 120a6e
                              ctrl_ports, partLevel, 1, (int)frame, 1, 0, 0, 0,
Shinya Kitaoka 120a6e
                              0, lastframe, getIdentifier());
Shinya Kitaoka 120a6e
  } else if (TRaster64P raster64 = tile.getRaster()) {
Shinya Kitaoka 120a6e
    TFlash *flash = 0;
Shinya Kitaoka 120a6e
    myEngine.render_particles(flash, &tile, part_ports, ri, p_size, p_offset,
Shinya Kitaoka 120a6e
                              ctrl_ports, partLevel, 1, (int)frame, 1, 0, 0, 0,
Shinya Kitaoka 120a6e
                              0, lastframe, getIdentifier());
Shinya Kitaoka 120a6e
  } else
Shinya Kitaoka 120a6e
    throw TException("ParticlesFx: unsupported Pixel Type");
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void ParticlesFx::compute(TFlash &flash, int frame) {
Shinya Kitaoka 120a6e
  // Particles is currently disabled in Flash...
Shinya Kitaoka 120a6e
  return;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void ParticlesFx::compatibilityTranslatePort(int major, int minor,
Shinya Kitaoka 120a6e
                                             std::string &portName) {
Shinya Kitaoka 120a6e
  VersionNumber version(major, minor);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  if (version < VersionNumber(1, 16)) {
Shinya Kitaoka 120a6e
    if (portName == "Texture") portName = "Texture1";
Shinya Kitaoka 120a6e
  } else if (version < VersionNumber(1, 20)) {
Shinya Kitaoka 120a6e
    int idx;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    bool chop =
Shinya Kitaoka 120a6e
        ((idx = portName.find("Texture")) != std::string::npos && idx > 0) ||
Shinya Kitaoka 120a6e
        ((idx = portName.find("Control")) != std::string::npos && idx > 0);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    if (chop) portName.erase(portName.begin(), portName.begin() + idx);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//==============================================================================
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
FX_PLUGIN_IDENTIFIER(ParticlesFx, "particlesFx");