squareScaleManip/squareScaleManip.cpp

squareScaleManip/squareScaleManip.cpp
//-
// ==========================================================================
// Copyright 1995,2006,2008 Autodesk, Inc. All rights reserved.
//
// Use of this software is subject to the terms of the Autodesk
// license agreement provided at the time of installation or download,
// or which otherwise accompanies this software in either electronic
// or hard copy form.
// ==========================================================================
//+
//
// squareScaleManip.cpp
// Creates manipulator node squareScaleManipulator
// Creates command squareManipCmd
//
// This example demonstrates how to use the MPxManipulatorNode
// class along with a command to create a user defined
// manipulator. The manipulator created is a simple square
// with the 4 sides as OpenGL pickable components. As you
// move the pickable component, selected transforms have
// their scale attribute modified.
// A corresponding command is used to create and delete
// the manipulator node and to support undo/redo etc.
/*
// To show this example using MEL, run the following:
loadPlugin squareScaleManip.so;
squareManipCmd -create;
// To delete the manipulator using MEL:
squareManipCmd -delete;
*/
#include "squareScaleManip.h"
#include <maya/MIOStream.h>
#include <maya/MMatrix.h>
#include <maya/MVector.h>
#include <maya/MSelectionList.h>
#include <maya/MItSelectionList.h>
#include <maya/MFnTransform.h>
#include <maya/MGlobal.h>
#include <maya/MFnCamera.h>
#include <maya/MTemplateCommand.h>
#include <maya/MDrawRegistry.h>
#include <maya/MDrawContext.h>
#include <maya/MUserData.h>
#include <maya/MStateManager.h>
// Statics
MTypeId squareScaleManipulator::id( 0x81046 );
MString squareScaleManipulator::registrantId("SquareScaleManipPlugin");
const MPoint squareScaleManipulator::topLeft(-0.5f, 0.5f, 0.0f);
const MPoint squareScaleManipulator::topRight(0.5f, 0.5f, 0.0f);
const MPoint squareScaleManipulator::bottomLeft(-0.5f, -0.5f, 0.0f);
const MPoint squareScaleManipulator::bottomRight(0.5f, -0.5f, 0.0f);
//
// class implementation
//
squareScaleManipulator::squareScaleManipulator()
, fActiveName(0)
, fTopName(0)
, fRightName(0)
, fBottomName(0)
, fLeftName(0)
, fMousePointGlName(MPoint())
{
// Populate initial points
topLeft.get(fTopLeft);
topRight.get(fTopRight);
bottomLeft.get(fBottomLeft);
bottomRight.get(fBottomRight);
// Setup the plane with a point on the
// plane along with a normal
MPoint pointOnPlane(topLeft);
// Normal = cross product of two vectors on the plane
MVector normalToPlane =
(MVector(topLeft) - MVector(topRight)) ^
(MVector(topRight) - MVector(bottomRight));
// Necessary to normalize
normalToPlane.normalize();
// Plane defined by a point and a normal
fPlane.setPlane(pointOnPlane, normalToPlane);
}
squareScaleManipulator::~squareScaleManipulator()
{
}
void squareScaleManipulator::postConstructor()
{
// Get the starting value of the pickable items
MGLuint glPickableItem;
glFirstHandle(glPickableItem);
fTopName = glPickableItem++;
fBottomName = glPickableItem++;
fLeftName = glPickableItem++;
fRightName = glPickableItem++;
}
//virtual
void squareScaleManipulator::preDrawUI( const M3dView &view )
{
M3dView *viewPtr = const_cast<M3dView*>( &view );
MDagPath dpath;
viewPtr->getCamera(dpath);
fDrawManip = shouldDraw(dpath);
}
//virtual
void squareScaleManipulator::drawUI(MHWRender::MUIDrawManager& drawManager, const MHWRender::MFrameContext& frameContext) const
{
if(!fDrawManip)
return;
const short defaultCol = mainColor();
setHandleColor(drawManager, fTopName, defaultCol);
drawManager.line(MPoint(fTopLeft), MPoint(fTopRight));
drawManager.endDrawable();
// In Viewport 2.0, the bottom line is not selectable
MPoint center = (MPoint(fBottomLeft) + MPoint(fBottomRight))*0.5;
drawManager.beginDrawable();
drawManager.setColor(MColor(1.0f, 0.0f, 0.0f, 1.0f));
drawManager.text(center, MString("The bottom line is not selectable"));
drawManager.endDrawable();
setHandleColor(drawManager, fBottomName, defaultCol);
drawManager.line(MPoint(fBottomLeft), MPoint(fBottomRight));
drawManager.endDrawable();
setHandleColor(drawManager, fLeftName, defaultCol);
drawManager.line(MPoint(fTopLeft), MPoint(fBottomLeft));
drawManager.endDrawable();
setHandleColor(drawManager, fRightName, defaultCol);
drawManager.line(MPoint(fTopRight), MPoint(fBottomRight));
drawManager.endDrawable();
}
// virtual
void squareScaleManipulator::draw(
M3dView &view,
const MDagPath &path,
{
// Are we in the right view
MDagPath dpath;
view.getCamera(dpath);
if (!shouldDraw(dpath))
{
return;
}
// do the draw using common helper method
view.beginGL();
// Place before you draw the manipulator component that can
// be pickable.
colorAndName(view, fTopName, true, mainColor());
glBegin(GL_LINES);
glVertex3fv(fTopLeft);
glVertex3fv(fTopRight);
glEnd();
// Right
colorAndName(view, fRightName, true, mainColor());
glBegin(GL_LINES);
glVertex3fv(fTopRight);
glVertex3fv(fBottomRight);
glEnd();
// Bottom
colorAndName(view, fBottomName, true, mainColor());
glBegin(GL_LINES);
glVertex3fv(fBottomRight);
glVertex3fv(fBottomLeft);
glEnd();
// Left
colorAndName(view, fLeftName, true, mainColor());
glBegin(GL_LINES);
glVertex3fv(fBottomLeft);
glVertex3fv(fTopLeft);
glEnd();
view.endGL();
}
// virtual
MStatus squareScaleManipulator::doPress(M3dView& view)
{
// Reset the mousePoint information on a new press
fMousePointGlName = MPoint::origin;
updateDragInformation();
return MS::kSuccess;
}
// virtual
MStatus squareScaleManipulator::doDrag(M3dView& view)
{
updateDragInformation();
return MS::kSuccess;
}
// virtual
MStatus squareScaleManipulator::doRelease(M3dView& view)
{
// Scale nodes on the selection list. Implementation
// is very simple and will not support undo.
for (MItSelectionList iter(list); !iter.isDone(); iter.next())
{
MObject node;
MStatus status;
iter.getDependNode(node);
MFnTransform xform(node, &status);
if (MS::kSuccess == status)
{
double newScale[3];
newScale[0] = fMousePointGlName.x + 1;
newScale[1] = fMousePointGlName.y + 1;
newScale[2] = fMousePointGlName.z + 1;
xform.setScale(newScale);
}
}
return MS::kSuccess;
}
MStatus squareScaleManipulator::updateDragInformation()
{
// Find the mouse point in local space
MPoint localMousePoint;
MVector localMouseDirection;
if (MS::kFailure == mouseRay(localMousePoint, localMouseDirection))
return MS::kFailure;
// Find the intersection of the mouse point with the
// manip plane
MPoint mouseIntersectionWithManipPlane;
if (!fPlane.intersect(localMousePoint, localMouseDirection, mouseIntersectionWithManipPlane))
return MS::kFailure;
fMousePointGlName = mouseIntersectionWithManipPlane;
if (glActiveName(fActiveName))
{
// Reset draw points
topLeft.get(fTopLeft);
topRight.get(fTopRight);
bottomLeft.get(fBottomLeft);
bottomRight.get(fBottomRight);
float* start = 0;
float* end = 0;
if (fActiveName == fTopName)
{
start = fTopLeft;
end = fTopRight;
}
if (fActiveName == fBottomName)
{
start = fBottomLeft;
end = fBottomRight;
}
if (fActiveName == fRightName)
{
start = fTopRight;
end = fBottomRight;
}
if (fActiveName == fLeftName)
{
start = fTopLeft;
end = fBottomLeft;
}
if (start && end)
{
// Find a vector on the plane
lineMath line;
MPoint a(start[0], start[1], start[2]);
MPoint b(end[0], end[1], end[2]);
MPoint vab = a - b;
// Define line with a point and a vector on the plane
line.setLine(start, vab);
MPoint cpt;
// Find the closest point so that we can get the
// delta change of the mouse in local space
if (line.closestPoint(fMousePointGlName, cpt))
{
fMousePointGlName.x -= cpt.x;
fMousePointGlName.y -= cpt.y;
fMousePointGlName.z -= cpt.z;
}
// Move draw points
start[0] += (float)fMousePointGlName.x;
start[1] += (float)fMousePointGlName.y;
start[2] +=(float) fMousePointGlName.z;
end[0] += (float)fMousePointGlName.x;
end[1] += (float)fMousePointGlName.y;
end[2] +=(float) fMousePointGlName.z;
}
}
return MS::kSuccess;
}
bool squareScaleManipulator::shouldDraw(const MDagPath& cameraPath) const
{
MStatus status;
MFnCamera camera(cameraPath, &status);
if (!status)
{
return false;
}
const char* nameBuffer = camera.name().asChar();
if (0 == nameBuffer)
{
return false;
}
if ((0 == strstr(nameBuffer,"persp")) &&
(0 == strstr(nameBuffer,"front")))
{
return false;
}
return true;
}
//
// Template command that creates and deletes the manipulator
//
class squareManipCmd;
char cmdName[] = "squareManipCmd";
char nodeName[] = "squareScaleManipulator";
class squareManipCmd : public MTemplateCreateNodeCommand<squareManipCmd,cmdName,nodeName>
{
public:
//
squareManipCmd()
{}
};
static squareManipCmd _squareManipCmd;
//
// Static methods
//
void* squareScaleManipulator::creator()
{
return new squareScaleManipulator();
}
MStatus squareScaleManipulator::initialize()
{
// No-op
return MS::kSuccess;
}
//
// Entry points
//
MStatus initializePlugin( MObject obj )
{
MStatus status;
MFnPlugin plugin( obj, PLUGIN_COMPANY, "2009", "Any");
status = plugin.registerNode(
nodeName,
squareScaleManipulator::id,
squareScaleManipulator::creator,
squareScaleManipulator::initialize,
if (!status) {
status.perror("registerNode");
return status;
}
status = _squareManipCmd.registerCommand( obj );
if (!status) {
status.perror("registerCommand");
return status;
}
return status;
}
MStatus uninitializePlugin( MObject obj )
{
MStatus status;
MFnPlugin plugin( obj );
status = plugin.deregisterNode( squareScaleManipulator::id );
if (!status) {
status.perror("deregisterNode");
return status;
}
status = _squareManipCmd.deregisterCommand( obj );
if (!status) {
status.perror("deregisterCommand");
return status;
}
return status;
}