jitterNode.cpp
#include <string.h>
#include <stdio.h>
#include <maya/MIOStream.h>
#include <maya/MPxNode.h>
#include <maya/MString.h>
#include <maya/MTypeId.h>
#include <maya/MPlug.h>
#include <maya/MFnNumericAttribute.h>
#include <maya/MVector.h>
#include <maya/MFnPlugin.h>
#include <maya/MDataBlock.h>
#include <maya/MDataHandle.h>
static unsigned int xRand, yRand, zRand;
float randomd()
{
float result;
unsigned int a = xRand/177, b = xRand%177;
unsigned int c = yRand/176, d = yRand%176;
unsigned int e = zRand/178, f = zRand%178;
xRand = (171 * b - 2 * a) % 30269;
yRand = (172 * d - 35 * c) % 30307;
zRand = (170 * f - 63 * e) % 30323;
result = (float) xRand/30269.0f + yRand/30307.0f + zRand/30323.0f;
return result - ((int) result);
}
void seedd(unsigned char nx, unsigned char ny, unsigned char nz)
{
xRand = nx;
yRand = ny;
zRand = nz;
randomd();
randomd();
randomd();
randomd();
}
class jitter : public MPxNode
{
public:
jitter();
virtual ~jitter();
virtual MStatus compute( const MPlug& plug, MDataBlock& data );
static void* creator();
static MStatus initialize();
static MObject time;
static MObject scale;
static MObject input;
static MObject output;
static MTypeId id;
};
MTypeId jitter::id( 0x80009 );
MObject jitter::time;
MObject jitter::scale;
MObject jitter::input;
MObject jitter::output;
void* jitter::creator()
{
return new jitter();
}
MStatus jitter::initialize()
{
MFnNumericAttribute nAttr;
MStatus stat;
time = nAttr.create( "time", "tm", MFnNumericData::kFloat, 0.0 );
nAttr.setStorable(true);
scale = nAttr.create( "scale", "sc", MFnNumericData::kFloat, 1.0 );
nAttr.setStorable(true);
input = nAttr.create( "input", "in", MFnNumericData::kFloat, 0.0 );
nAttr.setStorable(true);
output = nAttr.create( "output", "out", MFnNumericData::kFloat, 0.0 );
nAttr.setWritable(false);
nAttr.setStorable(false);
stat = addAttribute( time );
if (!stat) { stat.perror("addAttribute"); return stat;}
stat = addAttribute( scale );
if (!stat) { stat.perror("addAttribute"); return stat;}
stat = addAttribute( input );
if (!stat) { stat.perror("addAttribute"); return stat;}
stat = addAttribute( output );
if (!stat) { stat.perror("addAttribute"); return stat;}
stat = attributeAffects( time, output );
if (!stat) { stat.perror("attributeAffects"); return stat;}
stat = attributeAffects( scale, output );
if (!stat) { stat.perror("attributeAffects"); return stat;}
stat = attributeAffects( input, output );
if (!stat) { stat.perror("attributeAffects"); return stat;}
return MS::kSuccess;
}
jitter::jitter() {}
jitter::~jitter() {}
MStatus jitter::compute( const MPlug& plug, MDataBlock& data )
{
MStatus returnStatus;
if( plug == output )
{
MDataHandle timeData = data.inputValue( time, &returnStatus );
MDataHandle scaleData = data.inputValue( scale, &returnStatus );
MDataHandle inputData = data.inputValue( input, &returnStatus );
if( returnStatus != MS::kSuccess )
cerr << "ERROR getting data\n";
else
{
float currentFrame = timeData.asFloat();
float scaleFactor = scaleData.asFloat();
float inValue = inputData.asFloat();
unsigned char seed = (unsigned char)currentFrame;
seedd( seed, seed * 17, seed * 23 );
float tmp = randomd();
float result = ( tmp - 0.5f ) * scaleFactor + inValue;
MDataHandle outHandle = data.outputValue( jitter::output );
outHandle.set( result );
data.setClean(plug);
}
}
else {
return MS::kUnknownParameter;
}
return MS::kSuccess;
}
MStatus initializePlugin( MObject obj )
{
MStatus status;
MFnPlugin plugin( obj, PLUGIN_COMPANY, "3.0", "Any");
status = plugin.registerNode( "jitter", jitter::id,
jitter::creator, jitter::initialize );
if (!status) {
status.perror("registerNode");
return status;
}
return status;
}
MStatus uninitializePlugin( MObject obj)
{
MStatus status;
MFnPlugin plugin( obj );
status = plugin.deregisterNode( jitter::id );
if (!status) {
status.perror("deregisterNode");
return status;
}
return status;
}