407 lines
14 KiB

/*==LICENSE==*
CyanWorlds.com Engine - MMOG client, server and tools
Copyright (C) 2011 Cyan Worlds, Inc.
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
You can contact Cyan Worlds, Inc. by email legal@cyan.com
or by snail mail at:
Cyan Worlds, Inc.
14617 N Newport Hwy
Mead, WA 99021
*==LICENSE==*/
#include "cyPhysics.h"
#include "pyKey.h"
#include "pyGeometry3.h"
#include "pyMatrix44.h"
#include <Python.h>
// glue functions
PYTHON_CLASS_DEFINITION(ptPhysics, cyPhysics);
PYTHON_DEFAULT_NEW_DEFINITION(ptPhysics, cyPhysics)
PYTHON_DEFAULT_DEALLOC_DEFINITION(ptPhysics)
PYTHON_INIT_DEFINITION(ptPhysics, args, keywords)
{
PYTHON_RETURN_INIT_OK;
}
PYTHON_METHOD_DEFINITION(ptPhysics, netForce, args)
{
char forceFlag;
if (!PyArg_ParseTuple(args, "b", &forceFlag))
{
PyErr_SetString(PyExc_TypeError, "netForce requires a boolean argument");
PYTHON_RETURN_ERROR;
}
self->fThis->SetNetForce(forceFlag != 0);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, enable, args)
{
char stateFlag = 1;
if (!PyArg_ParseTuple(args, "|b", &stateFlag))
{
PyErr_SetString(PyExc_TypeError, "enable expects an optional boolean argument");
PYTHON_RETURN_ERROR;
}
self->fThis->EnableT(stateFlag != 0);
PYTHON_RETURN_NONE;
}
PYTHON_BASIC_METHOD_DEFINITION(ptPhysics, disable, Disable)
PYTHON_BASIC_METHOD_DEFINITION(ptPhysics, disableCollision, DisableCollision)
PYTHON_BASIC_METHOD_DEFINITION(ptPhysics, enableCollision, EnableCollision)
PYTHON_METHOD_DEFINITION(ptPhysics, warp, args)
{
PyObject *positionObject = NULL;
if (!PyArg_ParseTuple(args, "O", &positionObject))
{
PyErr_SetString(PyExc_TypeError, "warp expects a ptPoint3 or ptMatrix44 object");
PYTHON_RETURN_ERROR;
}
if (pyPoint3::Check(positionObject))
{
pyPoint3 *pos = pyPoint3::ConvertFrom(positionObject);
self->fThis->Warp(*pos);
PYTHON_RETURN_NONE;
}
else if (pyMatrix44::Check(positionObject))
{
pyMatrix44 *mat = pyMatrix44::ConvertFrom(positionObject);
self->fThis->WarpMat(*mat);
PYTHON_RETURN_NONE;
}
PyErr_SetString(PyExc_TypeError, "warp expects a ptPoint3 or ptMatrix44 object");
PYTHON_RETURN_ERROR;
}
PYTHON_METHOD_DEFINITION(ptPhysics, warpObj, args)
{
PyObject *keyObject = NULL;
if (!PyArg_ParseTuple(args, "O", &keyObject))
{
PyErr_SetString(PyExc_TypeError, "warpObj expects a ptKey");
PYTHON_RETURN_ERROR;
}
if (!pyKey::Check(keyObject))
{
PyErr_SetString(PyExc_TypeError, "warpObj expects a ptKey");
PYTHON_RETURN_ERROR;
}
pyKey *key = pyKey::ConvertFrom(keyObject);
self->fThis->WarpObj(*key);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, move, args)
{
PyObject *directionObject = NULL;
float distance;
if (!PyArg_ParseTuple(args, "Of", &directionObject, &distance))
{
PyErr_SetString(PyExc_TypeError, "move expects a ptVector3 and float");
PYTHON_RETURN_ERROR;
}
if (!pyVector3::Check(directionObject))
{
PyErr_SetString(PyExc_TypeError, "move expects a ptVector3 and float");
PYTHON_RETURN_ERROR;
}
pyVector3 *direction = pyVector3::ConvertFrom(directionObject);
self->fThis->Move(*direction, distance);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, rotate, args)
{
float radians;
PyObject *axisObject = NULL;
if (!PyArg_ParseTuple(args, "fO", &radians, &axisObject))
{
PyErr_SetString(PyExc_TypeError, "rotate expects a float and ptVector3");
PYTHON_RETURN_ERROR;
}
if (!pyVector3::Check(axisObject))
{
PyErr_SetString(PyExc_TypeError, "rotate expects a float and ptVector3");
PYTHON_RETURN_ERROR;
}
pyVector3 *axis = pyVector3::ConvertFrom(axisObject);
self->fThis->Rotate(radians, *axis);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, force, args)
{
PyObject *forceObject = NULL;
if (!PyArg_ParseTuple(args, "O", &forceObject))
{
PyErr_SetString(PyExc_TypeError, "force expects a ptVector3");
PYTHON_RETURN_ERROR;
}
if (!pyVector3::Check(forceObject))
{
PyErr_SetString(PyExc_TypeError, "force expects a ptVector3");
PYTHON_RETURN_ERROR;
}
pyVector3 *force = pyVector3::ConvertFrom(forceObject);
self->fThis->Force(*force);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, forceWithOffset, args)
{
PyObject *forceObject = NULL;
PyObject *offsetObject = NULL;
if (!PyArg_ParseTuple(args, "OO", &forceObject, &offsetObject))
{
PyErr_SetString(PyExc_TypeError, "forceWithOffset expects a ptVector3 and a ptPoint3");
PYTHON_RETURN_ERROR;
}
if ((!pyVector3::Check(forceObject)) || (!pyPoint3::Check(offsetObject)))
{
PyErr_SetString(PyExc_TypeError, "forceWithOffset expects a ptVector3 and a ptPoint3");
PYTHON_RETURN_ERROR;
}
pyVector3 *force = pyVector3::ConvertFrom(forceObject);
pyPoint3 *offset = pyPoint3::ConvertFrom(offsetObject);
self->fThis->ForceWithOffset(*force, *offset);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, torque, args)
{
PyObject *torqueObject = NULL;
if (!PyArg_ParseTuple(args, "O", &torqueObject))
{
PyErr_SetString(PyExc_TypeError, "torque expects a ptVector3");
PYTHON_RETURN_ERROR;
}
if (!pyVector3::Check(torqueObject))
{
PyErr_SetString(PyExc_TypeError, "torque expects a ptVector3");
PYTHON_RETURN_ERROR;
}
pyVector3 *torque = pyVector3::ConvertFrom(torqueObject);
self->fThis->Torque(*torque);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, impulse, args)
{
PyObject *forceObject = NULL;
if (!PyArg_ParseTuple(args, "O", &forceObject))
{
PyErr_SetString(PyExc_TypeError, "impulse expects a ptVector3");
PYTHON_RETURN_ERROR;
}
if (!pyVector3::Check(forceObject))
{
PyErr_SetString(PyExc_TypeError, "impulse expects a ptVector3");
PYTHON_RETURN_ERROR;
}
pyVector3 *force = pyVector3::ConvertFrom(forceObject);
self->fThis->Impulse(*force);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, impulseWithOffset, args)
{
PyObject *forceObject = NULL;
PyObject *offsetObject = NULL;
if (!PyArg_ParseTuple(args, "OO", &forceObject, &offsetObject))
{
PyErr_SetString(PyExc_TypeError, "impulseWithOffset expects a ptVector3 and a ptPoint3");
PYTHON_RETURN_ERROR;
}
if ((!pyVector3::Check(forceObject)) || (!pyPoint3::Check(offsetObject)))
{
PyErr_SetString(PyExc_TypeError, "impulseWithOffset expects a ptVector3 and a ptPoint3");
PYTHON_RETURN_ERROR;
}
pyVector3 *force = pyVector3::ConvertFrom(forceObject);
pyPoint3 *offset = pyPoint3::ConvertFrom(offsetObject);
self->fThis->ImpulseWithOffset(*force, *offset);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, angularImpulse, args)
{
PyObject *forceObject = NULL;
if (!PyArg_ParseTuple(args, "O", &forceObject))
{
PyErr_SetString(PyExc_TypeError, "angularImpulse expects a ptVector3");
PYTHON_RETURN_ERROR;
}
if (!pyVector3::Check(forceObject))
{
PyErr_SetString(PyExc_TypeError, "angularImpulse expects a ptVector3");
PYTHON_RETURN_ERROR;
}
pyVector3 *force = pyVector3::ConvertFrom(forceObject);
self->fThis->AngularImpulse(*force);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, damp, args)
{
float damp;
if (!PyArg_ParseTuple(args, "f", &damp))
{
PyErr_SetString(PyExc_TypeError, "damp expects a float");
PYTHON_RETURN_ERROR;
}
self->fThis->Damp(damp);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, shiftMass, args)
{
PyObject *offestObject = NULL;
if (!PyArg_ParseTuple(args, "O", &offestObject))
{
PyErr_SetString(PyExc_TypeError, "shiftMass expects a ptVector3");
PYTHON_RETURN_ERROR;
}
if (!pyVector3::Check(offestObject))
{
PyErr_SetString(PyExc_TypeError, "shiftMass expects a ptVector3");
PYTHON_RETURN_ERROR;
}
pyVector3 *offset = pyVector3::ConvertFrom(offestObject);
self->fThis->ShiftMass(*offset);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, suppress, args)
{
char doSuppress;
if (!PyArg_ParseTuple(args, "b", &doSuppress))
{
PyErr_SetString(PyExc_TypeError, "suppress expects a boolean");
PYTHON_RETURN_NONE;
}
self->fThis->Suppress(doSuppress != 0);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, setLinearVelocity, args)
{
PyObject *velocity = NULL;
if (!PyArg_ParseTuple(args, "O", &velocity))
{
PyErr_SetString(PyExc_TypeError, "setVelocity expects a ptVector3");
PYTHON_RETURN_ERROR;
}
if (!pyVector3::Check(velocity))
{
PyErr_SetString(PyExc_TypeError, "setVelocity expects a ptVector3");
PYTHON_RETURN_ERROR;
}
pyVector3 *velocityVec = pyVector3::ConvertFrom(velocity);
self->fThis->SetLinearVelocity(*velocityVec);
PYTHON_RETURN_NONE;
}
PYTHON_METHOD_DEFINITION(ptPhysics, setAngularVelocity, args)
{
PyObject *velocity = NULL;
if (!PyArg_ParseTuple(args, "O", &velocity))
{
PyErr_SetString(PyExc_TypeError, "setAngularVelocity expects a ptVector3");
PYTHON_RETURN_ERROR;
}
if (!pyVector3::Check(velocity))
{
PyErr_SetString(PyExc_TypeError, "setAngularVelocity expects a ptVector3");
PYTHON_RETURN_ERROR;
}
pyVector3 *velocityVec = pyVector3::ConvertFrom(velocity);
self->fThis->SetAngularVelocity(*velocityVec);
PYTHON_RETURN_NONE;
}
PYTHON_START_METHODS_TABLE(ptPhysics)
PYTHON_METHOD(ptPhysics, netForce, "Params: forceFlag\nSpecify whether this object needs to use messages that are forced to the network\n"
"- This is to be used if your Python program is running on only one client\n"
"Such as a game master, only running on the client that owns a particular object"),
PYTHON_METHOD(ptPhysics, enable, "Params: state=1\nSets the physics enable state for the sceneobject attached"),
PYTHON_BASIC_METHOD(ptPhysics, disable, "Disables physics on the sceneobject attached"),
PYTHON_BASIC_METHOD(ptPhysics, disableCollision, "Disables collision detection on the attached sceneobject"),
PYTHON_BASIC_METHOD(ptPhysics, enableCollision, "Enables collision detection on the attached sceneobject"),
PYTHON_METHOD(ptPhysics, warp, "Params: position\nWarps the sceneobject to a specified location.\n"
"'position' can be a ptPoint3 or a ptMatrix44"),
PYTHON_METHOD(ptPhysics, warpObj, "Params: objkey\nWarps the sceneobject to match the location and orientation of the specified object"),
PYTHON_METHOD(ptPhysics, move, "Params: direction,distance\nMoves the attached sceneobject the specified distance in the specified direction"),
PYTHON_METHOD(ptPhysics, rotate, "Params: radians,axis\nRotates the attached sceneobject the specified radians around the specified axis"),
PYTHON_METHOD(ptPhysics, force, "Params: forceVector\nApplies the specified force to the attached sceneobject"),
PYTHON_METHOD(ptPhysics, forceWithOffset, "Params: forceVector,offsetPt\nApplies the specified offsetted force to the attached sceneobject"),
PYTHON_METHOD(ptPhysics, torque, "Params: torqueVector\nApplies the specified torque to the attached sceneobject"),
PYTHON_METHOD(ptPhysics, impulse, "Params: impulseVector\nAdds the given vector to the attached sceneobject's velocity"),
PYTHON_METHOD(ptPhysics, impulseWithOffset, "Params: impulseVector,offsetPt\nAdds the given vector to the attached sceneobject's velocity\n"
"with the specified offset"),
PYTHON_METHOD(ptPhysics, angularImpulse, "Params: impulseVector\nAdd the given vector (representing a rotation axis and magnitude) to\n"
"the attached sceneobject's velocity"),
PYTHON_METHOD(ptPhysics, damp, "Params: damp\nReduce all velocities on the object (0 = all stop, 1 = no effect)"),
PYTHON_METHOD(ptPhysics, shiftMass, "Params: offsetVector\nShifts the attached sceneobject's center to mass in the specified direction and distance"),
PYTHON_METHOD(ptPhysics, suppress, "Params: doSuppress\nCompletely remove the physical, but keep it around so it\n"
"can be added back later."),
PYTHON_METHOD(ptPhysics, setLinearVelocity, "Params: velocityVector\nSets the objects LinearVelocity to the specified vector"),
PYTHON_METHOD(ptPhysics, setAngularVelocity, "Params: velocityVector\nSets the objects AngularVelocity to the specified vector"),
PYTHON_END_METHODS_TABLE;
// Type structure definition
PLASMA_DEFAULT_TYPE(ptPhysics, "Plasma physics class");
// required functions for PyObject interoperability
PyObject *cyPhysics::New(PyObject *sender, PyObject *recvr)
{
ptPhysics *newObj = (ptPhysics*)ptPhysics_type.tp_new(&ptPhysics_type, NULL, NULL);
if (sender != NULL)
{
pyKey *senderKey = pyKey::ConvertFrom(sender);
newObj->fThis->SetSender(senderKey->getKey());
}
if (recvr != NULL)
{
pyKey *recvrKey = pyKey::ConvertFrom(recvr);
newObj->fThis->AddRecvr(recvrKey->getKey());
}
newObj->fThis->SetNetForce(false);
return (PyObject*)newObj;
}
PYTHON_CLASS_CHECK_IMPL(ptPhysics, cyPhysics)
PYTHON_CLASS_CONVERT_FROM_IMPL(ptPhysics, cyPhysics)
///////////////////////////////////////////////////////////////////////////
//
// AddPlasmaClasses - the python module definitions
//
void cyPhysics::AddPlasmaClasses(PyObject *m)
{
PYTHON_CLASS_IMPORT_START(m);
PYTHON_CLASS_IMPORT(m, ptPhysics);
PYTHON_CLASS_IMPORT_END(m);
}