代码拉取完成,页面将自动刷新
/*
\file wrapper.cpp
\remarks Implementation of the ValueWrapper and ObjectWrapper classes
\author Blur Studio (c) 2010
\email beta@blur.com
\license This software is released under the GNU General Public License. For more info, visit: http://www.gnu.org/
*/
#include "imports.h"
#include <string>
#include <sstream> // for std::stringstream
#include "macros.h"
#include "wrapper.h"
#include "protector.h"
#include "structmember.h"
#include "Parser.h" // for print functions
// define these properties if the MAKE_PY26_COMPAT preprocessor definition exists
#ifdef MAKE_PY26_COMPAT
typedef inquiry lenfunc;
typedef int Py_ssize_t;
typedef intobjargproc ssizeobjargproc;
typedef intargfunc ssizeargfunc;
#endif
#ifndef UNICODE
// For some reason python doesn't actually use the active code page
// that it gets from windows for [En|De]codeMBC functionality.
// So we have to get it from python and use it for the encoding/decoding
// of data coming from max.
static const char * getEncoding()
{
static char codePage[50] = {0};
if( codePage[0] == 0 ) {
PyObject * locale_mod = PyImport_ImportModule("locale");
PyObject * enc_func = locale_mod ? PyObject_GetAttrString(locale_mod,"getpreferredencoding") : 0;
PyObject * enc = PyObject_CallFunction(enc_func,NULL);
if( enc && PyString_Check(enc) )
strncpy(codePage,PyString_AsString(enc),49);
Py_XDECREF(enc);
Py_XDECREF(enc_func);
Py_XDECREF(locale_mod);
}
return codePage;
}
#endif
PyStringToMCHAR::PyStringToMCHAR( PyObject * pyString, bool stealRef )
: mObject( 0 )
#ifndef UNICODE
, mData( 0 )
#endif
{
#ifdef UNICODE
if( PyString_Check(pyString) ) {
mObject = PyUnicode_DecodeUTF8( PyString_AsString(pyString), PyString_Size(pyString), NULL );
if( stealRef )
Py_DECREF(pyString);
} else if( PyUnicode_Check(pyString) ) {
mObject = pyString;
if( !stealRef )
Py_INCREF(mObject);
}
#else
if( pyString ) {
if( PyString_Check(pyString) ) {
mObject = pyString;
if( !stealRef )
Py_XINCREF(mObject);
mData = PyString_AsString(mObject);
} else if( PyUnicode_Check(pyString) ) {
mObject = PyUnicode_AsEncodedString(pyString,getEncoding(),NULL);
if( stealRef )
Py_DECREF(pyString);
mData = PyString_AsString(mObject);
}
}
#endif
}
PyStringToMCHAR::PyStringToMCHAR( const char * data )
: mObject( 0 )
#ifndef UNICODE
, mData( data )
#endif
{
#ifdef UNICODE
mObject = PyUnicode_DecodeUTF8( data, strlen(data), NULL );
if( !mObject ) {
PY_ERROR_PRINT_CLEAR
}
#endif
}
PyStringToMCHAR::~PyStringToMCHAR()
{
Py_XDECREF(mObject);
}
const MCHAR * PyStringToMCHAR::mchar()
{
#ifdef UNICODE
return mObject ? PyUnicode_AsUnicode(mObject) : 0;
#else
return mData;
#endif
}
MCharToPyString::MCharToPyString( const MCHAR * mchar )
: mObject( 0 )
, mUtf8Object( 0 )
, mMChars( mchar )
{
}
MCharToPyString::~MCharToPyString()
{
Py_XDECREF(mObject);
Py_XDECREF(mUtf8Object);
}
PyObject * MCharToPyString::pyString()
{
if( !mObject ) {
#ifdef UNICODE
mObject = PyUnicode_FromUnicode(mMChars,wcslen(mMChars));
#else
unsigned char * c = (unsigned char*)mMChars;
bool isAscii = false;
if( isAscii ) {
mObject = PyString_FromString(mMChars);
} else {
int len = _tcslen(mMChars);
mObject = PyUnicode_Decode(mMChars,len,getEncoding(),NULL);
if( !mObject ) {
PY_ERROR_PRINT_CLEAR
mObject = PyUnicode_DecodeUTF8(mMChars,len,NULL);
if( !mObject ) {
PY_ERROR_PRINT_CLEAR
mObject = PyUnicode_DecodeASCII(mMChars,len,"replace");
}
}
}
#endif
}
if( !mObject ) {
PY_ERROR_PRINT_CLEAR
mprintf( _T("%s\n"), _T("<critical error converting maxscript string to python, please contact IT>") );
mObject = PyString_FromString("<critical error converting maxscript string to python, please contact IT>");
}
return mObject;
}
PyObject * MCharToPyString::pyStringRef()
{
PyObject * ret = pyString();
Py_XINCREF(ret);
return ret;
}
const char * MCharToPyString::data()
{
if( mObject ) {
if( PyString_Check(mObject) )
return PyString_AsString(mObject);
}
#ifdef UNICODE
if( !mUtf8Object && mObject ) {
mUtf8Object = PyUnicode_AsUTF8String(mObject);
if( !mUtf8Object )
PyErr_Print();
}
return PyString_Check(mUtf8Object) ? PyString_AsString(mUtf8Object) : 0;
#else
return mMChars;
#endif
}
//---------------------------------------------------------------
// ValueWrapper Implementation
//---------------------------------------------------------------
// ctor
PyObject*
ValueWrapper_new( PyTypeObject* type, PyObject* args, PyObject* kwds ) {
ValueWrapper* self;
self = (ValueWrapper*)type->tp_alloc(type, 0);
self->mValue = NULL;
self->mNext = self->mPrev = 0;
return (PyObject*) self;
}
static int
ValueWrapper_init( PyObject *self, PyObject *args, PyObject *kwds ) {
// Since the ValueWrapper Python class is exposed via the Py3dsMax
// module, we need to make sure it is not directly instantiated. The
// reason for exposing it is to allow for isinstance checks and in
// general make it behave like a normal Python type/class.
PyErr_SetString(PyExc_NotImplementedError, "Py3dsMax.ValueWrapper may not be directly instantiated.");
return -1;
}
#ifndef PyMODINIT_FUNC
#define PyMODINIT_FUNC void
#endif
PyMODINIT_FUNC
ValueWrapper_init(void) {
ValueWrapperType.ob_type = &PyType_Type;
PyType_Ready( &ValueWrapperType );
}
static long
ValueWrapper_hash( ValueWrapper* self ) {
PyObject * pyAddress = PyLong_FromUnsignedLongLong( (unsigned PY_LONG_LONG)self );
long hash = PyObject_Hash(pyAddress);
Py_DECREF(pyAddress);
return hash;
}
// dtor
void
ValueWrapper_dealloc( ValueWrapper* self ) {
Protector::unprotect( self );
self->ob_type->tp_free( (PyObject*) self );
}
// __call__ function: called when calling a function on a ValueWrapper instance
PyObject*
ValueWrapper_call( ValueWrapper* self, PyObject* args, PyObject* kwds ) {
// Step 1: calculate how many Value**'s we are going to need to pass the Value* method
Py_ssize_t arg_count = (args) ? PyTuple_Size( args ) : 0;
Py_ssize_t key_count = (kwds) ? PyDict_Size( kwds ) : -1;
Py_ssize_t mxs_count = (key_count == -1) ? arg_count : arg_count + 1 + (key_count*2);
// Step 2: protect the maxcript memory we are going to use
MXS_PROTECT( three_typed_value_locals( Value* method, Value* result, StringStream* log ) );
// Step 3: pull out the proper method from maxscript
MXS_EVAL( self->mValue, vl.method );
// override the current log for error catching purposes
vl.log = new StringStream();
CharStream* old_log = thread_local(current_stdout);
((MAXScript_TLS*)TlsGetValue(thread_locals_index))->current_stdout = vl.log;
vl.result = NULL;
// Step 4: call the maxscript method
if ( vl.method ) {
// Step 5: check to see if we need maxscript arguments
if ( mxs_count ) {
// Step 6: if we do, we need to create a volatile array and protect it
Value** mxs_args;
value_local_array(mxs_args,mxs_count);
// Step 7: add converted arguments
for ( int i = 0; i < arg_count; i++ ) {
mxs_args[i] = ObjectWrapper::intern( PyTuple_GetItem( args, i ) );
}
// Step 8: add converted keywords
if ( key_count > 0 ) {
PyObject *pkey, *pvalue;
Py_ssize_t pos = 0;
int key_pos = 0;
// Maxscript deliminates keywords from arguments by supplying the &keyarg_marker pointer
// for when the keywords start, and from there does a key, value pairing in the array
mxs_args[arg_count] = &keyarg_marker;
while ( PyDict_Next(kwds, &pos, &pkey, &pvalue ) ) {
{
PyStringToMCHAR mchar(pkey);
mxs_args[ arg_count + 1 + (key_pos*2) ] = Name::intern( mchar.mchar() );
}
mxs_args[ arg_count + 2 + (key_pos*2) ] = ObjectWrapper::intern( pvalue );
key_pos++;
}
// Release the key and value pointers
pkey = NULL;
pvalue = NULL;
}
// Step 9: call the method and use try/catch to protect the memory
try {
vl.result = vl.method->apply( mxs_args, mxs_count )->get_heap_ptr();
}
catch ( MAXScriptException& e ) {
//show_source_pos();
StringStream* buffer = new StringStream(_T("Maxscript error has occurred while calling function: ") );
buffer->puts(vl.log->to_string());
const MCHAR * filename = thread_local(source_file) ? thread_local(source_file)->to_string() : 0;
buffer->printf(_T("-- %s %i %i\n"), filename ? filename : _T("unknown file"), thread_local(source_pos), thread_local(source_line) );
e.sprin1(buffer);
MCharToPyString pybuf(buffer->to_string());
PyErr_SetObject( PyExc_RuntimeError, pybuf.pyString() );
restore_current_frames();
clear_error_source_data();
MAXScript_signals = 0;
return 0;
}
catch ( ... ) {
MXS_CLEARERRORS();
PyErr_SetString( PyExc_RuntimeError, "Unknown MAXScript Error Occurred" );
}
/* catch ( ... ) {
MXS_CLEARERRORS();
MCharToPyString pylog(vl.log->to_string());
PyErr_SetString( PyExc_RuntimeError, pylog.data() );
vl.result = NULL;
} */
// clear the value local memory
pop_value_local_array(mxs_args);
}
else {
// Step 6: if we don't, simply call the method with a NULL array
try { vl.result = vl.method->apply( NULL, 0 ); }
catch ( ... ) {
MXS_CLEARERRORS();
MCharToPyString pylog(vl.log->to_string());
PyErr_SetObject( PyExc_RuntimeError, pylog.pyString() );
vl.result = NULL;
}
}
}
// restore the old log
((MAXScript_TLS*)TlsGetValue(thread_locals_index))->current_stdout = old_log;
// Step 10: convert the result
PyObject* output = NULL;
if ( vl.result )
output = ObjectWrapper::py_intern( vl.result );
else if( !PyErr_Occurred() ) {
output = Py_None;
Py_INCREF(Py_None);
}
// Step 11: cleanup the maxscript errors
MXS_CLEANUP();
return output;
}
// __cmp__ function: called when comparing 2 ValueWrapper instances (a == b, a < b, a <= b, etc.)
int
ValueWrapper_compare( PyObject* self, PyObject* other ) {
int result = -1;
// Step 1: protect the maxscript memory
MXS_PROTECT( two_value_locals( mxs_self, mxs_other ) );
// Step 2: convert the two items to values
MXS_EVAL( ObjectWrapper::intern( self ), vl.mxs_self );
MXS_EVAL( ObjectWrapper::intern( other ), vl.mxs_other );
// Step 3: make sure both items exist
if ( vl.mxs_self && vl.mxs_other ) {
// Step 4: compare direct pointers
if ( vl.mxs_self == vl.mxs_other || vl.mxs_self->eval() == vl.mxs_other->eval() )
result = 0;
else {
// Step 5: use maxscript's built in <, ==, > checks
try {
if ( vl.mxs_self->eq_vf( &vl.mxs_other, 1 ) == &true_value ) { result = 0; }
else if ( vl.mxs_self->lt_vf( &vl.mxs_other, 1 ) == &true_value ) { result = -1; }
else if ( vl.mxs_self->gt_vf( &vl.mxs_other, 1 ) == &true_value ) { result = 1; }
}
catch ( ... ) {
MXS_CLEARERRORS();
}
}
}
// Step 6: cleanup maxscript memory
MXS_CLEANUP();
return result;
}
// __str__ function: convert a Value* wrapper to a string
PyObject*
ValueWrapper_str( ValueWrapper* self ) {
// Step 1: protect values
MXS_PROTECT( two_typed_value_locals( Value* mxs_check, StringStream* mxs_stream ) );
// Step 2: evaluate the value
MXS_EVAL( self->mValue, vl.mxs_check );
// Step 3: check for simple conversions
PyObject* output = NULL;
if ( is_name( vl.mxs_check ) || is_string( vl.mxs_check ) ) {
output = MCharToPyString(vl.mxs_check->to_string()).pyStringRef();
}
// Step 4: block recursive struct printouts
else if ( is_struct( vl.mxs_check ) )
output = PyString_FromString( "<mxs.StructDef instance>" );
// Step 5: try to use the builtin maxscript print out
else {
vl.mxs_stream = new StringStream();
try {
vl.mxs_check->sprin1( vl.mxs_stream );
output = MCharToPyString(vl.mxs_stream->to_string()).pyStringRef();
}
MXS_CATCHERRORS();
}
// Step 6: cleanup maxscript memory
MXS_CLEANUP();
if( !output ) {
output = Py_None;
Py_INCREF(output);
}
return output;
}
// __getattr__ function: get a property by name from a Value* instance
PyObject*
ValueWrapper_getattr( ValueWrapper* self, char* key ) {
// check for pre-defined python properties
PyObject* tmp;
PyObject* check = PyString_FromString(key);
if ( !(tmp = PyObject_GenericGetAttr( (PyObject*) self, check )) ) {
Py_XDECREF(check);
// check for python internals
if ( key[0] == '_' || !PyErr_ExceptionMatches(PyExc_AttributeError) ) {
return NULL;
}
PyErr_Clear();
}
else {
Py_XDECREF(check);
return tmp;
}
// return the default maxscript value
one_value_local(result);
PyStringToMCHAR mkey(PyString_FromString(key),true);
try { vl.result = ((ValueWrapper*) self)->mValue->eval()->_get_property( Name::intern(mkey.mchar()) ); }
catch ( ... ) {
PyObject * self_str = ValueWrapper_str(self);
PyErr_Format( PyExc_AttributeError, "%s is not a property of %s", key, PyString_AsString(self_str) );
Py_DECREF(self_str);
pop_value_locals();
return 0;
}
tmp = ObjectWrapper::py_intern( vl.result );
pop_value_locals();
return tmp;
}
// __setattr__ function: sets a property by name for a Value*
int
ValueWrapper_setattr( ValueWrapper* self, char* key, PyObject* value ) {
bool success = true;
PyStringToMCHAR mkey(PyString_FromString(key),true);
try { self->mValue->eval()->_set_property( Name::intern(mkey.mchar()), ObjectWrapper::intern(value) ); }
catch ( ... ) { success = false; }
if ( success ) {
return 0;
}
else {
PyObject* pkey = PyString_FromString(key);
int ret = PyObject_GenericSetAttr( (PyObject*) self, pkey, value );
Py_DECREF(pkey);
return ret;
}
}
//------- Sequence Methods
// __len__ function: calculate the length of a maxscript value
int
ValueWrapper_length( PyObject* self ) {
// Step 1: protect the maxscript memory
MXS_PROTECT( one_value_local( mxs_check ) );
// Step 2: evaluate the value
MXS_EVAL( ((ValueWrapper*) (self))->mValue, vl.mxs_check );
int count = NULL;
// Step 3: use the count property to check if an object has length in maxscript
try { count = vl.mxs_check->_get_property( n_count )->to_int(); }
MXS_CATCHERRORS();
// Step 4: cleanup maxscript memory
MXS_CLEANUP();
return count;
}
// __getitem__ function: get an item from an index in a maxscript value
PyObject*
ValueWrapper_item( PyObject* self, int index ) {
MXS_PROTECT( three_value_locals( mxs_check, mindex, result ) );
MXS_EVAL( ((ValueWrapper*) self)->mValue, vl.mxs_check );
int count = 0;
// pull the lenght of the collection
try { count = vl.mxs_check->_get_property( n_count )->to_int(); }
catch ( ... ) {
MXS_CLEARERRORS();
PyErr_SetString( PyExc_IndexError, "__getitem__ cannot access item for index" );
MXS_CLEANUP();
return NULL;
}
// support pythonic negative lookup
if ( index < 0 ) {
index = count + index;
}
// make sure the index is valid
if ( !( 0 <= index && index < count) ) {
PyErr_SetString( PyExc_IndexError, "__getitem__ index is out of range" );
MXS_CLEANUP();
return NULL;
}
// pull the value
vl.mindex = Integer::intern( index + 1 );
try { vl.result = vl.mxs_check->get_vf( &vl.mindex, 1 ); }
MXS_CATCHERRORS();
PyObject* output = NULL;
if ( !vl.result ) { PyErr_SetString( PyExc_IndexError, "__getitem__ index is out of range" ); }
else { output = ObjectWrapper::py_intern(vl.result); }
MXS_CLEANUP();
return output;
}
// __getitem__ function: get an item based on an abstract python object
PyObject*
ValueWrapper_objitem( PyObject* self, PyObject* key ) {
// Step 1: check to see if the key is a number, then use the above method
if ( key->ob_type == &PyInt_Type ) {
return ValueWrapper_item( self, PyInt_AsLong( key ) );
}
// Step 2: protect the maxscript memory
MXS_PROTECT( three_value_locals( mxs_index, mxs_check, mxs_result ) );
// Step 3: evaluate the value
MXS_EVAL( ((ValueWrapper*) (self))->mValue, vl.mxs_check );
// Step 4: convert the python object to a maxscript key
vl.mxs_index = ObjectWrapper::intern( key );
// Step 5: try to pull the value from the sequence using virtual functions
try { vl.mxs_result = vl.mxs_check->get_vf( &vl.mxs_index, 1 ); }
MXS_CATCHERRORS();
// Step 6: convert the maxscript result to a python value
PyObject* output = NULL;
if ( vl.mxs_result ) { output = ObjectWrapper::py_intern( vl.mxs_result ); }
else { PyErr_SetString( PyExc_IndexError, "__getitem__ could not get a maxscript value" ); }
// Step 7: cleanup maxscript memory
MXS_CLEANUP();
return output;
}
// __setitem__ function: set a maxscript value by an index
int
ValueWrapper_setitem( PyObject* self, int index, PyObject* value ) {
// Step 1: protect maxscript memory
MXS_PROTECT( two_value_locals( mxs_check, mxs_result ) );
// Step 2: evaluate the value
MXS_EVAL( ((ValueWrapper*) (self))->mValue, vl.mxs_check );
// Step 3: calculate the length of the sequence
int count = 0;
try { count = vl.mxs_check->_get_property( n_count )->to_int(); }
MXS_CATCHERRORS();
if ( count ) {
// Step 3: resolve index differences in maxscript & python
if ( index < 0 )
index += count;
if ( 0 <= index && index < count ) {
// Step 4: create a volatile array of values to pass to the put virtual function
Value** arg_list;
value_local_array( arg_list, 2 );
arg_list[0] = Integer::intern( index + 1 );
arg_list[1] = ObjectWrapper::intern( value );
try { vl.mxs_result = vl.mxs_check->put_vf( arg_list, 2 ); }
MXS_CATCHERRORS();
// Step 5: pop the local array
pop_value_local_array(arg_list);
}
}
// Step 6: convert the result
int result = -1;
if ( vl.mxs_result ) { result = 0; }
else { PyErr_SetString( PyExc_IndexError, "__setitem__ index is out of range" ); }
// Step 6: cleanup maxscript memory
MXS_CLEANUP();
return result;
}
// __setitem__ function: set a maxscript value by python object
int
ValueWrapper_setobjitem( PyObject* self, PyObject* key, PyObject* value ) {
// Step 1: check to see if the python object is a number type, if so, use the above method
if ( key->ob_type == &PyInt_Type ) {
return ValueWrapper_setitem( self, PyInt_AsLong( key ), value );
}
// Step 2: protect the maxscript memory
MXS_PROTECT( one_value_local( mxs_check ) );
// Step 3: evaluate the value
MXS_EVAL( ((ValueWrapper*) (self))->mValue, vl.mxs_check );
// Step 4: create a volatile maxscript value array to pass to the put virtual function
Value** arg_list;
value_local_array( arg_list, 2 );
arg_list[0] = ObjectWrapper::intern( key );
arg_list[1] = ObjectWrapper::intern( value );
// Step 5: call the put virtual method
int result = -1;
try {
vl.mxs_check->put_vf( arg_list, 2 );
result = 0;
}
MXS_CATCHERRORS();
// Step 6: convert the results
if ( result == -1 ) { PyErr_SetString( PyExc_IndexError, "__setitem__ could not set the maxscript value" ); }
// Step 7: cleanup maxscript memory
pop_value_local_array( arg_list );
MXS_CLEANUP();
return result;
}
// Number Methods
// __add__ function: called when trying to add two wrapper values together
PyObject*
ValueWrapper_add( PyObject* self, PyObject* other ) {
// Step 1: protect maxscript memory
MXS_PROTECT( three_value_locals( mxs_self, mxs_other, mxs_value ) );
// Step 2: convert the two items to values
MXS_EVAL( ObjectWrapper::intern( self ), vl.mxs_self );
MXS_EVAL( ObjectWrapper::intern( other ), vl.mxs_other );
// Step 3: calculate maxscript value
try { vl.mxs_value = vl.mxs_self->plus_vf( &vl.mxs_other, 1 ); }
MXS_CATCHERRORS();
// Step 4: convert to python
PyObject* output = NULL;
if ( vl.mxs_value ) { output = ObjectWrapper::py_intern( vl.mxs_value ); }
else { PyErr_SetString( PyExc_ArithmeticError, "__add__ error: could not add maxscript values together" ); }
// Step 5: cleanup maxscipt memory
MXS_CLEANUP();
return output;
}
// __sub__ function: called when trying to subtract two wrapper values together
PyObject*
ValueWrapper_subtract( PyObject* self, PyObject* other ) {
// Step 1: protect maxscript memory
MXS_PROTECT( three_value_locals( mxs_self, mxs_other, mxs_value ) );
// Step 2: convert the two items to values
MXS_EVAL( ObjectWrapper::intern( self ), vl.mxs_self );
MXS_EVAL( ObjectWrapper::intern( other ), vl.mxs_other );
// Step 3: calculate maxscript value
try { vl.mxs_value = vl.mxs_self->minus_vf( &vl.mxs_other, 1 ); }
MXS_CATCHERRORS();
// Step 4: convert to python
PyObject* output = NULL;
if ( vl.mxs_value ) { output = ObjectWrapper::py_intern( vl.mxs_value ); }
else { PyErr_SetString( PyExc_ArithmeticError, "__sub__ error: could not add maxscript values together" ); }
// Step 5: cleanup maxscipt memory
MXS_CLEANUP();
return output;
}
// __div__ function: called when trying to divide two wrapper values together
PyObject*
ValueWrapper_divide( PyObject* self, PyObject* other ) {
// Step 1: protect maxscript memory
MXS_PROTECT( three_value_locals( mxs_self, mxs_other, mxs_value ) );
// Step 2: convert the two items to values
MXS_EVAL( ObjectWrapper::intern( self ), vl.mxs_self );
MXS_EVAL( ObjectWrapper::intern( other ), vl.mxs_other );
// Step 3: calculate maxscript value
try { vl.mxs_value = vl.mxs_self->div_vf( &vl.mxs_other, 1 ); }
MXS_CATCHERRORS();
// Step 4: convert to python
PyObject* output = NULL;
if ( vl.mxs_value ) { output = ObjectWrapper::py_intern( vl.mxs_value ); }
else { PyErr_SetString( PyExc_ArithmeticError, "__div__ error: could not add maxscript values together" ); }
// Step 5: cleanup maxscipt memory
MXS_CLEANUP();
return output;
}
// __mul__ function: called when trying to multiply two wrapper values together
PyObject*
ValueWrapper_multiply( PyObject* self, PyObject* other ) {
// Step 1: protect maxscript memory
MXS_PROTECT( three_value_locals( mxs_self, mxs_other, mxs_value ) );
// Step 2: convert the two items to values
MXS_EVAL( ObjectWrapper::intern( self ), vl.mxs_self );
MXS_EVAL( ObjectWrapper::intern( other ), vl.mxs_other );
// Step 3: calculate maxscript value
try { vl.mxs_value = vl.mxs_self->times_vf( &vl.mxs_other, 1 ); }
MXS_CATCHERRORS();
// Step 4: convert to python
PyObject* output = NULL;
if ( vl.mxs_value ) { output = ObjectWrapper::py_intern( vl.mxs_value ); }
else { PyErr_SetString( PyExc_ArithmeticError, "__mul__ error: could not add maxscript values together" ); }
// Step 5: cleanup maxscipt memory
MXS_CLEANUP();
return output;
}
// __pow__ function: called when trying to raise one number by the other
PyObject*
ValueWrapper_power( PyObject* self, PyObject* other, PyObject *args ) {
// Step 1: protect maxscript memory
MXS_PROTECT( three_value_locals( mxs_self, mxs_other, mxs_value ) );
// Step 2: convert the two items to values
MXS_EVAL( ObjectWrapper::intern( self ), vl.mxs_self );
MXS_EVAL( ObjectWrapper::intern( other ), vl.mxs_other );
// Step 3: calculate maxscript value
try { vl.mxs_value = vl.mxs_self->pwr_vf( &vl.mxs_other, 1 ); }
MXS_CATCHERRORS();
// Step 4: convert to python
PyObject* output = NULL;
if ( vl.mxs_value ) { output = ObjectWrapper::py_intern( vl.mxs_value ); }
else { PyErr_SetString( PyExc_ArithmeticError, "__pow__ error: could not add maxscript values together" ); }
// Step 5: cleanup maxscipt memory
MXS_CLEANUP();
return output;
}
// __abs__ function: called when trying to divide two wrapper values together
PyObject*
ValueWrapper_absolute( PyObject* self ) {
// Step 1: protect maxscript memory
MXS_PROTECT( two_value_locals( mxs_self, mxs_value ) );
// Step 2: convert the two items to values
MXS_EVAL( ObjectWrapper::intern( self ), vl.mxs_self );
// Step 3: calculate maxscript value
try { vl.mxs_value = vl.mxs_self->abs_vf( NULL, 0 ); }
MXS_CATCHERRORS();
// Step 4: convert to python
PyObject* output = NULL;
if ( vl.mxs_value ) { output = ObjectWrapper::py_intern( vl.mxs_value ); }
else { PyErr_SetString( PyExc_ArithmeticError, "__abs__ error: could not add maxscript values together" ); }
// Step 5: cleanup maxscipt memory
MXS_CLEANUP();
return output;
}
// __int__ function: convert an object to an integer
PyObject*
ValueWrapper_int( PyObject* self ) {
// Step 1: protect maxscript memory
MXS_PROTECT( one_value_local( mxs_self ) );
// Step 2: convert the two items to values
MXS_EVAL( ObjectWrapper::intern( self ), vl.mxs_self );
// Step 3: calculate maxscript value
int result = 0;
try { result = vl.mxs_self->to_int(); }
MXS_CATCHERRORS();
// Step 4: convert to python
PyObject* output = PyInt_FromLong( result );
// Step 5: cleanup maxscipt memory
MXS_CLEANUP();
return output;
}
// __float__ function: convert an object to an integer
PyObject*
ValueWrapper_float( PyObject* self ) {
// Step 1: protect maxscript memory
MXS_PROTECT( one_value_local( mxs_self ) );
// Step 2: convert the two items to values
MXS_EVAL( ObjectWrapper::intern( self ), vl.mxs_self );
// Step 3: calculate maxscript value
float result = 0;
try { result = vl.mxs_self->to_float(); }
MXS_CATCHERRORS();
// Step 4: convert to python
PyObject* output = PyFloat_FromDouble( result );
// Step 5: cleanup maxscipt memory
MXS_CLEANUP();
return output;
}
// __neg__ function: return the negative of a wrapper object
PyObject*
ValueWrapper_negative( PyObject* self ) {
// Step 1: create the multiplier
PyObject* mult = PyInt_FromLong(-1);
PyObject* result = ValueWrapper_multiply( self, mult );
// Step 2: free the multiplier
Py_DECREF( mult );
return result;
}
// __nonzero__ function: return true always
int
ValueWrapper_nonzero( PyObject* self ) {
return 1;
}
// create custom methods
static PyObject*
ValueWrapper_property( PyObject* self, PyObject* args ) {
PyObject * ret = 0;
char* propName;
if ( !PyArg_ParseTuple( args, "s", &propName ) )
return NULL;
one_value_local(result);
PyStringToMCHAR mpropName(PyString_FromString(propName),true);
try { vl.result = ((ValueWrapper*) self)->mValue->eval()->_get_property( Name::intern(mpropName.mchar()) ); }
catch ( ... ) {
PyObject * self_str = ValueWrapper_str((ValueWrapper*)self);
PyErr_Format( PyExc_AttributeError, "%s is not a property of %s", propName, PyString_AsString(self_str) );
Py_DECREF(self_str);
}
if ( vl.result ) {
ret = ObjectWrapper::py_intern( vl.result );
}
pop_value_locals();
return ret;
}
static PyObject*
ValueWrapper_setProperty( PyObject* self, PyObject* args ) {
char* propName;
PyObject* propValue;
if ( !PyArg_ParseTuple( args, "sO", &propName, &propValue ) )
return NULL;
bool success = true;
PyStringToMCHAR mpropName(PyString_FromString(propName),true);
try { ((ValueWrapper*) self)->mValue->eval()->_set_property( Name::intern(mpropName.mchar()), ObjectWrapper::intern(propValue) ); }
catch ( ... ) { success = false; }
if ( success ) {
Py_INCREF(Py_True);
return Py_True;
}
else {
Py_INCREF(Py_False);
return Py_False;
}
}
//--------------------------------------------------------------
// create the python mappings
// define the custom methods for a wrapper
PyMethodDef ValueWrapper_methods[] = {
// windows methods
{ "property", (PyCFunction)ValueWrapper_property, METH_VARARGS, "Get a property from a MXS wrapper" },
{ "setProperty", (PyCFunction)ValueWrapper_setProperty, METH_VARARGS, "Set a property from a MXS wrapper" },
{ NULL, NULL, 0, NULL }
};
// sequence methods
static PySequenceMethods proxy_as_sequence = {
(lenfunc) ValueWrapper_length, // sq_length
0, // sq_concat
0, // sq_repeat
(ssizeargfunc) ValueWrapper_item, // sq_item
0, // sq_slice
(ssizeobjargproc) ValueWrapper_setitem, // sq_ass_item
0, // sq_ass_slice
0, // sq_contains
0, // sq_inplace_concat
0, // sq_inplace_repeat
};
// mapping methods
static PyMappingMethods proxy_as_mapping = {
(lenfunc) ValueWrapper_length, // mp_length
(binaryfunc) ValueWrapper_objitem, // mp_subscript
(objobjargproc) ValueWrapper_setobjitem, // mp_ass_subscript
};
// number methods
static PyNumberMethods proxy_as_number = {
(binaryfunc) ValueWrapper_add, // nb_add
(binaryfunc) ValueWrapper_subtract, // nb_subtract
(binaryfunc) ValueWrapper_multiply, // nb_multiply
(binaryfunc) ValueWrapper_divide, // nb_divide
0, // nb_remainder
0, // nb_divmod
(ternaryfunc) ValueWrapper_power, // nb_power
(unaryfunc) ValueWrapper_negative, // nb_negative
0, // nb_positive
(unaryfunc) ValueWrapper_absolute, // nb_absolute
(inquiry) ValueWrapper_nonzero, // nb_nonzero
0, // nb_invert
0, // nb_lshift
0, // nb_rshift
0, // nb_and
0, // nb_xor
0, // nb_or
0, // nb_coerce
(unaryfunc) ValueWrapper_int, // nb_int
0, // nb_long
(unaryfunc) ValueWrapper_float, // nb_float,
0, // nb_oct
0, // nb_hex
0, // nb_inplace_add
0, // nb_inplace_subtract
0, // nb_inplace_multiply
0, // nb_inplace_divide
0, // nb_inplace_remainder
0, // nb_inplace_power
0, // nb_inplace_lshift
0, // nb_inplace_rshift
0, // nb_inplace_and
0, // nb_inplace_xor
0, // nb_inplace_or
};
// python type methods
PyTypeObject ValueWrapperType = {
PyObject_HEAD_INIT(NULL)
0, // ob_size
"Py3dsMax.ValueWrapper", // tp_name
sizeof(ValueWrapper), // tp_basicsize
0, // tp_itemsize
(destructor)ValueWrapper_dealloc, // tp_dealloc
0, // tp_print
(getattrfunc)ValueWrapper_getattr, // tp_getattr
(setattrfunc)ValueWrapper_setattr, // tp_setattr
(cmpfunc)ValueWrapper_compare, // tp_compare
0, // tp_repr
&proxy_as_number, // tp_as_number
&proxy_as_sequence, // tp_as_sequence
&proxy_as_mapping, // tp_as_mapping
(hashfunc)ValueWrapper_hash, // tp_hash
(ternaryfunc)ValueWrapper_call, // tp_call
(reprfunc)ValueWrapper_str, // tp_str
0, // tp_getattro
0, // tp_setattro
0, // tp_as_buffer
Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE | Py_TPFLAGS_CHECKTYPES, // tp_flags
"Maxscript Value Wrapper", // tp_doc
0, // tp_traverse
0, // tp_clear
0, // tp_richcompare
0, // tp_weaklistoffset
0, // tp_iter
0, // tp_iternext
ValueWrapper_methods, // tp_methods
0, // tp_members
0, // tp_getset
0, // tp_base
0, // tp_dict
0, // tp_descr_get
0, // tp_descr_set
offsetof(ValueWrapper, dict), // tp_dictoffset
ValueWrapper_init, // tp_init
0, // tp_alloc
ValueWrapper_new, // tp_new
};
struct TypedFloat {
PyFloatObject fo;
bool isFloat;
};
void TypedFloatType_init(void)
{
TypedFloatType.tp_new = PyType_GenericNew;
TypedFloatType.tp_base = &PyFloat_Type;
PyType_Ready(&TypedFloatType);
Py_INCREF(&TypedFloatType);
}
static PyObject * buildTypedFloat(double d, bool isFloat)
{
PyObject * self = TypedFloatType.tp_alloc(&TypedFloatType, 0);
PyObject * args = PyTuple_New(2);
Py_INCREF(self);
PyTuple_SET_ITEM(args,0,(PyObject*)self);
PyTuple_SET_ITEM(args,1,PyFloat_FromDouble(d));
PyFloat_Type.tp_init(self, args, NULL);
Py_DECREF(args);
((TypedFloat*)self)->isFloat = isFloat;
return (PyObject*)self;
}
static int
TypedFloat_init(PyObject * self, PyObject *args, PyObject *kwds)
{
TypedFloat * tf = (TypedFloat*)self;
if (PyTuple_Size(args) >= 1) {
double d = PyFloat_AsDouble(PyTuple_GET_ITEM(args,0));
if (PyErr_Occurred())
return -1;
tf->fo.ob_fval = d;
}
tf->isFloat = (PyTuple_Size(args) >= 2) && (PyTuple_GET_ITEM(args,1) == Py_True);
return 0;
}
static PyMemberDef TypedFloat_members[] = {
{"isFloat", T_BOOL, offsetof(TypedFloat,isFloat), RO, "True if the TypedFloat is a float, False if it is a double" },
{NULL}
};
// python type methods
PyTypeObject TypedFloatType = {
PyObject_HEAD_INIT(NULL)
0, // ob_size
"Py3dsMax.TypedFloat", // tp_name
sizeof(TypedFloat), // tp_basicsize
0, // tp_itemsize
0, // tp_dealloc
0, // tp_print
0, // tp_getattr
0, // tp_setattr
0, // tp_compare
0, // tp_repr
0, // tp_as_number
0, // tp_as_sequence
0, // tp_as_mapping
0, // tp_hash
0, // tp_call
0, // tp_str
0, // tp_getattro
0, // tp_setattro
0, // tp_as_buffer
Py_TPFLAGS_DEFAULT, // tp_flags
"Typed(float vs double) Python Float", // tp_doc
0, // tp_traverse
0, // tp_clear
0, // tp_richcompare
0, // tp_weaklistoffset
0, // tp_iter
0, // tp_iternext
0, // tp_methods
TypedFloat_members, // tp_members
0, // tp_getset
0, // tp_base
0, // tp_dict
0, // tp_descr_get
0, // tp_descr_set
0, // tp_dictoffset
TypedFloat_init, // tp_init
0, // tp_alloc
0 // tp_new
};
//--------------------------------------------------------------------------------
// ObjectWrapper implementation
//--------------------------------------------------------------------------------
visible_class_instance( ObjectWrapper, "PyObjectWrapper" );
// ctor
ObjectWrapper::ObjectWrapper( PyObject* obj )
: mObject( obj )
, mObjectDict( 0 )
, mPyString( 0 )
{
PyGILState_STATE gstate;
gstate = PyGILState_Ensure();
tag = class_tag(ObjectWrapper);
Py_XINCREF(obj);
PyGILState_Release(gstate);
}
// dtor
ObjectWrapper::~ObjectWrapper()
{
PyGILState_STATE gstate;
gstate = PyGILState_Ensure();
Py_XDECREF( mObject );
Py_XDECREF( mObjectDict );
Py_XDECREF( mPyString );
PyGILState_Release(gstate);
}
// __call__ function: call a python object with maxscript values
Value*
ObjectWrapper::apply( Value** arg_list, int count, CallContext* cc ) {
// Step 1: Make sure this object is callable
if ( mObject && PyCallable_Check( mObject ) ) {
// Step 2; protect the maxscript memory
MXS_PROTECT( one_value_local( output ) );
vl.output = &undefined;
// Step 3: determine the keyarg_marker to deliminate the break between args and keywords
int py_count = 0;
for ( int i = 0; i < count; i++ ) {
// if the current item is the marker, then we have keywords
if ( arg_list[i] == &keyarg_marker )
break;
py_count++;
}
// Step 4: generate the python arguments and keywords
PyObject* args = NULL;
PyObject* kwds = NULL;
PyObject* py_result = NULL;
// Step 5: generate the function arguments (args cannot be NULL)
PyGILState_STATE gstate;
gstate = PyGILState_Ensure();
args = PyTuple_New( py_count );
for ( int i = 0; i < py_count; i++ )
PyTuple_SetItem( args, i, ObjectWrapper::py_intern( arg_list[i]->eval() ) );
// Step 6: generate the function keywords
if ( py_count != count ) {
kwds = PyDict_New();
for ( int i = py_count + 1; i < count; i += 2 ) {
MCharToPyString pystring( arg_list[i]->eval()->to_string() );
PyDict_SetItem( kwds, pystring.pyString(), ObjectWrapper::py_intern( arg_list[i+1]->eval() ) );
}
}
// Step 7: execture the python call
py_result = PyObject_Call( mObject, args, kwds );
if ( PyErr_Occurred() ) {
MXS_CLEARERRORS();
Py_XDECREF( args );
Py_XDECREF( kwds );
Py_XDECREF( py_result );
PY_ERROR_PRINT_THROW();
}
// Step 8: convert the result to a value
if ( py_result ) { vl.output = ObjectWrapper::intern( py_result ); }
else { mprintf( _T("Python Error: could not properly execute python function.\n") ); }
// Step 9: release the python memory
Py_XDECREF( args );
Py_XDECREF( kwds );
Py_XDECREF( py_result );
PyGILState_Release(gstate);
// Step 10: return the value, protecting maxscript memory
MXS_RETURN( vl.output );
}
else { return Value::apply( arg_list, count, cc ); }
}
// collect function: this function is run when the garbage collection determines this maxscript variable is ready to be destroyed
void
ObjectWrapper::collect() {
// Python object cleanup is done in the dtor
delete this;
}
// __eq__ function: checks to see if one objectwrapper is equal to another
Value*
ObjectWrapper::eq_vf( Value** arg_list, int count ) {
if ( is_objectwrapper( arg_list[0] ) && is_objectwrapper( arg_list[0] ) ) {
return ( ((ObjectWrapper*) arg_list[0])->object() == ((ObjectWrapper*) arg_list[1])->object() ) ? &true_value : &false_value;
}
return &false_value;
}
// __getattr__ function: get a property of a python value from maxscript
Value*
ObjectWrapper::get_property( Value** arg_list, int count ) {
// Step 1: setup the output value
MXS_PROTECT( one_value_local( output ) );
vl.output = &undefined;
PyGILState_STATE gstate;
gstate = PyGILState_Ensure();
do {
// Step 2: get default properties
if ( arg_list[0] == n_count ) {
if ( PyMapping_Check( mObject ) )
vl.output = Integer::intern( PyMapping_Length( mObject ) );
else if ( PySequence_Check( mObject ) )
vl.output = Integer::intern( PySequence_Length( mObject ) );
else
vl.output = Integer::intern( 0 );
break;
}
// Step 3: lookup by keyword
// try to access the attribute directly
const MCHAR * m_key = arg_list[0]->eval()->to_string();
MCharToPyString pystr(m_key);
if ( PyObject_HasAttr( mObject, pystr.pyString() ) ) {
PyObject * attr = PyObject_GetAttr( mObject, pystr.pyString() );
vl.output = ObjectWrapper::intern( attr, false );
Py_XDECREF( attr );
break;
}
// In order to avoid clashes with maxscript keywords we allow a trailing underscore to be silently removed
{
const char * kstring = pystr.data();
int keyLen = strlen(kstring);
if( kstring[keyLen-1] == '_' ) {
char * tmpKey = new char[keyLen];
strncpy(tmpKey, kstring, keyLen-1);
tmpKey[keyLen-1] = 0;
if ( PyObject_HasAttrString( mObject, tmpKey ) ) {
PyObject * attr = PyObject_GetAttrString( mObject, tmpKey );
vl.output = ObjectWrapper::intern( attr, false );
delete [] tmpKey;
Py_XDECREF( attr );
break;
}
delete [] tmpKey;
}
}
if ( mObjectDict == NULL ) {
// loop through the internal dictionary comparing the lowercase attributes to see if we have one
PyObject **dictptr = _PyObject_GetDictPtr(mObject);
if ( dictptr != NULL ) {
mObjectDict = *dictptr;
Py_XINCREF(mObjectDict);
}
}
if ( mObjectDict != NULL ) {
// These will be borrowed references during the loop
PyObject *key, *value;
Py_ssize_t pos = 0;
// because maxscript does not preserve the actual case sensitivity for the property, we have to check against
// all possible options comparing lower case sensitvity to the keys
// create a lowercase version of the maxscript key
TSTR mxskstring = TSTR(m_key);
mxskstring.toLower();
while ( PyDict_Next( mObjectDict, &pos, &key, &value ) ) {
// create a lowercase version of the python key
PyStringToMCHAR mkey(key);
TSTR pykstring = TSTR(mkey.mchar());
pykstring.toLower();
// return the first instance in the dictionary that has matching lowercase keys
if ( pykstring == mxskstring ) {
PyObject * attr = PyObject_GetAttr( mObject, key );
vl.output = ObjectWrapper::intern( attr, false );
Py_XDECREF( attr );
break;
}
}
}
} while (false);
PyGILState_Release(gstate);
// Step 4: return the value, protecting maxscript memory
MXS_RETURN( vl.output );
}
// __setattr__ function: set a python value from maxscript
Value*
ObjectWrapper::set_property( Value** arg_list, int count ) {
PyGILState_STATE gstate;
gstate = PyGILState_Ensure();
// Step 1: lookup by keyword
const MCHAR* kstring = arg_list[1]->eval()->to_string();
MCharToPyString pykstring(kstring);
//if ( PyObject_HasAttr( mObject, pykstring.pyString() ) )
PyObject_SetAttr( mObject, pykstring.pyString(), ObjectWrapper::py_intern( arg_list[0]->eval() ) );
PyGILState_Release(gstate);
return &ok;
}
// __getitem__ function: get a value from a python sequence/mapping from maxscript
Value*
ObjectWrapper::get_vf( Value** arg_list, int count ) {
MXS_PROTECT( two_value_locals( output, key ) );
PyObject* py_key = NULL;
PyObject* py_result = NULL;
// Step 1: convert the input keys
vl.key = arg_list[0]->eval();
PyGILState_STATE gstate;
gstate = PyGILState_Ensure();
try {
if( is_number(vl.key) )
py_key = PyInt_FromLong( vl.key->to_int() );
else
py_key = MCharToPyString(vl.key->to_string()).pyStringRef();
}
MXS_CATCHERRORS();
// Step 2: call the getItem method for the python object
if ( py_key && mObject ) {
py_result = PyObject_GetItem( mObject, py_key );
}
// Step 4: convert the return to a maxscript value
vl.output = ObjectWrapper::intern( py_result );
// Step 5: release the python memory
Py_XDECREF( py_key );
Py_XDECREF( py_result );
PyGILState_Release(gstate);
// Step 6: return the maxscript value, protecting its memory
MXS_RETURN( vl.output );
}
// object function: return the python object for this instance
PyObject*
ObjectWrapper::object() {
return mObject;
}
// __setitem__ function: sets an item for a python sequence/mapping type
Value*
ObjectWrapper::put_vf( Value** arg_list, int count ) {
// \todo: implement this method
return &ok;
}
// print function: prints this value to the screen
void
ObjectWrapper::sprin1( CharStream* s ) {
s->puts( to_string() );
}
// __str__ function: converts this item to a string
const MCHAR*
ObjectWrapper::to_string() {
if ( mObject ) {
const MCHAR * ret = 0;
PyGILState_STATE gstate;
gstate = PyGILState_Ensure();
#ifdef UNICODE
if( PyUnicode_Check( mObject ) )
ret = PyUnicode_AsUnicode( mObject );
#else
if( PyString_Check( mObject ) )
ret = PyString_AsString( mObject );
#endif
if( !ret && !mPyString ) {
mPyString = PyObject_Str( mObject );
#ifdef UNICODE
if( mPyString ) {
PyObject * tmp = PyUnicode_FromObject(mPyString);
Py_DECREF( mPyString );
mPyString = tmp;
}
#endif
if( !mPyString )
PyErr_Clear();
}
if( !ret && mPyString )
#ifdef UNICODE
ret = PyUnicode_AsUnicode( mPyString );
#else
ret = PyString_AsString( mPyString );
#endif
PyGILState_Release(gstate);
return ret;
}
return _T("<<python: error accessing wrapper object>>");
}
// Static methods
// intern method: create a Value* internal from a PyObject* instance
Value*
ObjectWrapper::intern( PyObject* obj, bool unwrap ) {
PyGILState_STATE gstate;
gstate = PyGILState_Ensure();
Value* ret = 0;
// Step 1: convert NULL or Py_None values
if ( !obj || obj == Py_None )
ret = &undefined;
// Step 2: convert ValueWrapper instances
else if ( obj->ob_type == &ValueWrapperType ) {
one_value_local( output );
vl.output = ((ValueWrapper*) obj)->mValue->eval();
PyGILState_Release(gstate);
return_value( vl.output );
}
// Step 3: convert strings/unicodes
else if ( obj->ob_type == &PyString_Type || obj->ob_type == &PyUnicode_Type ) {
one_value_local(output);
PyStringToMCHAR mobj(obj);
vl.output = new String(mobj.mchar());
PyGILState_Release(gstate);
return_value(vl.output);
}
// Step 4: convert integers/longs
else if ( obj->ob_type == &PyInt_Type || obj->ob_type == &PyLong_Type )
ret = Integer::intern( PyInt_AsLong( obj ) );
// Step 5.1: convert typed floats^M
else if (obj->ob_type == &TypedFloatType) {
double val = PyFloat_AsDouble( obj );
if (((TypedFloat*)obj)->isFloat)
ret = Float::intern(val);
else
ret = Double::intern(val);
}
// Step 5.2: convert normal float^M
else if ( obj->ob_type == &PyFloat_Type )
ret = Double::intern( PyFloat_AsDouble( obj ) );
// Step 6: convert boolean
else if ( obj->ob_type == &PyBool_Type )
ret = ( obj == Py_True ) ? &true_value : &false_value;
// Step 7: convert lists/tuples
else if ( obj->ob_type == &PyList_Type || obj->ob_type == &PyTuple_Type ) {
Py_ssize_t count = PyObject_Length(obj);
// Step 8: create a maxscript array of items
one_typed_value_local(Array* output);
vl.output = new Array(count);
PyObject* iterator = PyObject_GetIter(obj);
PyObject* item;
if ( iterator != NULL ) {
while ( item = PyIter_Next(iterator) ) {
vl.output->append( ObjectWrapper::intern( item ) );
Py_DECREF(item);
}
Py_DECREF(iterator);
if ( PyErr_Occurred() ) {
// propogate error
}
}
else {
// propogate error
}
PyGILState_Release(gstate);
return_value(vl.output);
}
else {
// Step 12: If this is a Python object that has a _nativePointer
// attribute, check to see if it is a ValueWrapperType object, and
// if it is use it as described in step 2. This allows for
// wrapping ValueWrapperType objects in pure Python, but have them
// directly usable via Py3dsMax.
if ( PyObject_HasAttrString(obj, "_nativePointer") && unwrap == true ) {
PyObject* nObj = PyObject_GetAttrString(obj, "_nativePointer");
if ( nObj != NULL && nObj->ob_type == &ValueWrapperType ) {
one_value_local( output );
vl.output = ((ValueWrapper*) nObj)->mValue->eval();
Py_DECREF(nObj);
PyGILState_Release(gstate);
return_value( vl.output );
}
}
// Step 13: create an ObjectWrapper instance
ret = new ObjectWrapper(obj);
}
PyGILState_Release(gstate);
return ret;
}
// initialize function: Initializes the PyObject* class
bool
ObjectWrapper::init() {
PyGILState_STATE gstate;
gstate = PyGILState_Ensure();
ValueWrapperType.ob_type = &PyType_Type;
bool ret = ( PyType_Ready( &ValueWrapperType ) < 0 ) ? false : true;
PyGILState_Release(gstate);
return ret;
}
void
ObjectWrapper::log( PyObject* obj ) {
PyGILState_STATE gstate;
gstate = PyGILState_Ensure();
PyObject * self_str = ValueWrapper_str((ValueWrapper*)obj);
PyStringToMCHAR mself(self_str);
mprintf( _T("%s\n"), mself.mchar() );
Py_DECREF(self_str);
PyGILState_Release(gstate);
}
// is_wrapper function: checks the type of the object to make sure its a ValueWrapper
bool
ObjectWrapper::is_wrapper( PyObject* obj ) {
return ( obj->ob_type == &ValueWrapperType ) ? true : false;
}
void
ObjectWrapper::handleMaxscriptError() {
MAXScriptException* e = thread_local(current_exception);
// process the current exception
if ( e ) {
one_typed_value_local( StringStream* buffer );
vl.buffer = new StringStream( _T("MAXScript Error has occurred: \n") );
e->sprin1(vl.buffer);
MCharToPyString pybuf(vl.buffer->to_string());
PyErr_SetObject( PyExc_RuntimeError, pybuf.pyString() );
pop_value_locals();
}
// set the exception to unknown
else {
PyErr_SetString( PyExc_RuntimeError, "An unknown MAXScript error has occurred" );
}
}
// py_intern function: converts a maxscript internal to a python internal
PyObject*
ObjectWrapper::py_intern( Value* val ) {
// Step 1: evaluate the value
Value* mxs_check = NULL;
MXS_EVAL( val, mxs_check );
PyObject* ret = 0;
PyGILState_STATE gstate;
gstate = PyGILState_Ensure();
// Step 2: check for NULL, &undefined, or &unsupplied values
if ( !mxs_check || mxs_check == &undefined || mxs_check == &unsupplied ) {
Py_INCREF( Py_None );
PyGILState_Release(gstate);
return Py_None;
}
// Step 3: check for ObjectWrappers
if ( is_objectwrapper(mxs_check) ) {
ret = ((ObjectWrapper*) mxs_check)->mObject;
Py_INCREF( ret );
}
// Step 4: check for strings
else if ( is_string( mxs_check ) ) {
ret = MCharToPyString(mxs_check->to_string()).pyStringRef();
}
// Step 5: check for integers
else if ( is_integer( mxs_check ) )
ret = PyInt_FromLong( mxs_check->to_int() );
else if ( is_integer64( mxs_check ) )
ret = PyLong_FromLongLong( mxs_check->to_int64() );
// Step 6: check for all other numbers
else if ( is_number( mxs_check ) )
ret = buildTypedFloat(mxs_check->to_double(), is_float(mxs_check));
// Step 7: check for ok/true values
else if ( mxs_check == &ok || mxs_check == &true_value ) {
Py_INCREF( Py_True );
ret = Py_True;
}
// Step 8: check for false values
else if ( mxs_check == &false_value ) {
Py_INCREF( Py_False );
ret = Py_False;
}
// map object sets or arrays to a python array using a node_map
else if ( is_objectset( mxs_check ) || is_array( mxs_check ) ) {
ret = PyList_New(0);
Value* args[2] = { NULL, (Value*)ret };
node_map m = { NULL, ObjectWrapper::collectionMapper, args, 2 };
mxs_check->map( m );
}
// Step 9: check for all collections (except bitarrays, modifiers, and objectsets)
else if ( is_collection( mxs_check ) && !(is_bitarray( mxs_check ) || is_maxmodifierarray( mxs_check )) ) {
// Step 10: grab the collection's count
int count = mxs_check->_get_property( n_count )->to_int();
// Step 11: create output array and maxscript index
ret = PyList_New(count);
Value* index;
for ( int i = 0; i < count; i++ ) {
// Step 12: set the maxscript index to the count + 1 (maxscript is 1 based)
index = Integer::intern(i+1);
PyList_SetItem( ret, i, ObjectWrapper::py_intern( mxs_check->get_vf( &index, 1 ) ) );
}
}
else {
// Step 13: create a new ValueWrapper instance
ValueWrapper* output = (ValueWrapper*) ValueWrapper_new( &ValueWrapperType, NULL, NULL );
one_value_local(heap_ptr);
vl.heap_ptr = val->get_heap_ptr();
output->mValue = vl.heap_ptr;
Protector::protect( output );
pop_value_locals();
ret =(PyObject*) output;
}
if (ret == 0) {
mprintf( _T("Unknown maxscript type passed to py_intern\n") );
Py_INCREF(Py_None);
ret = Py_None;
}
PyGILState_Release(gstate);
return ret;
}
Value*
ObjectWrapper::collectionMapper( Value** arg_list, int count ) {
// method used to map items from an array to a python array - faster than using the get_vf method for non-array's
PyObject* collection = (PyObject*)arg_list[1];
PyObject* object = ObjectWrapper::py_intern( arg_list[0] );
PyList_Append( collection, object );
Py_DECREF(object);
return &ok;
}
HMODULE ObjectWrapper::hInstance = 0;
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。