| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921 | //-----------------------------------------------------------------------------// Copyright (c) 2012 GarageGames, LLC//// Permission is hereby granted, free of charge, to any person obtaining a copy// of this software and associated documentation files (the "Software"), to// deal in the Software without restriction, including without limitation the// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or// sell copies of the Software, and to permit persons to whom the Software is// furnished to do so, subject to the following conditions://// The above copyright notice and this permission notice shall be included in// all copies or substantial portions of the Software.//// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS// IN THE SOFTWARE.//-----------------------------------------------------------------------------#include "platform/platform.h"#include "console/arrayObject.h"#include "console/consoleTypes.h"#include "console/engineAPI.h"#include "math/mMathFn.h"IMPLEMENT_CONOBJECT(ArrayObject);ConsoleDocClass( ArrayObject,   "@brief Data structure for storing indexed sequences of key/value pairs.\n\n"   "This is a powerful array class providing PHP style arrays in TorqueScript.\n\n"   "The following features are supported:<ul>\n"   "<li>array pointers: this allows you to move forwards or backwards through "   "the array as if it was a list, including jumping to the start or end.</li>\n"   "<li>sorting: the array can be sorted in either alphabetic or numeric mode, "   "on the key or the value, and in ascending or descending order</li>\n"   "<li>add/remove elements: elements can be pushed/popped from the start or "   "end of the array, or can be inserted/erased from anywhere in the middle</li>\n"   "<li>removal of duplicates: remove duplicate keys or duplicate values</li>\n"   "<li>searching: search the array and return the index of a particular key or "   "value</li>\n"   "<li>counting: count the number of instaces of a particular value or key in "   "the array, as well as the total number of elements</li>\n"   "<li>advanced features: array append, array crop and array duplicate</li>\n"   "</ul>\n\n"   "Array element keys and values can be strings or numbers\n\n"   "@ingroup Scripting");bool ArrayObject::smDecreasing = false;bool ArrayObject::smCaseSensitive = false;const char* ArrayObject::smCompareFunction;S32 QSORT_CALLBACK ArrayObject::_valueCompare( const void* a, const void* b ){   ArrayObject::Element *ea = (ArrayObject::Element *) (a);   ArrayObject::Element *eb = (ArrayObject::Element *) (b);   S32 result = smCaseSensitive ? dStrnatcmp(ea->value, eb->value) : dStrnatcasecmp(ea->value, eb->value);   return ( smDecreasing ? -result : result );}S32 QSORT_CALLBACK ArrayObject::_valueNumCompare( const void* a, const void* b ){   ArrayObject::Element *ea = (ArrayObject::Element *) (a);   ArrayObject::Element *eb = (ArrayObject::Element *) (b);   F32 aCol = dAtof(ea->value);   F32 bCol = dAtof(eb->value);   F32 result = aCol - bCol;   S32 res = result < 0 ? -1 : (result > 0 ? 1 : 0);   return ( smDecreasing ? -res : res );}S32 QSORT_CALLBACK ArrayObject::_keyCompare( const void* a, const void* b ){   ArrayObject::Element *ea = (ArrayObject::Element *) (a);   ArrayObject::Element *eb = (ArrayObject::Element *) (b);   S32 result = smCaseSensitive ? dStrnatcmp(ea->key, eb->key) : dStrnatcasecmp(ea->key, eb->key);   return ( smDecreasing ? -result : result );}S32 QSORT_CALLBACK ArrayObject::_keyNumCompare( const void* a, const void* b ){   ArrayObject::Element *ea = (ArrayObject::Element *) (a);   ArrayObject::Element *eb = (ArrayObject::Element *) (b);   const char* aCol = ea->key;   const char* bCol = eb->key;   F32 result = dAtof(aCol) - dAtof(bCol);   S32 res = result < 0 ? -1 : (result > 0 ? 1 : 0);   return ( smDecreasing ? -res : res );}S32 QSORT_CALLBACK ArrayObject::_keyFunctionCompare( const void* a, const void* b ){   ArrayObject::Element* ea = ( ArrayObject::Element* )( a );   ArrayObject::Element* eb = ( ArrayObject::Element* )( b );      ConsoleValue cValue = Con::executef((const char*)smCompareFunction, ea->key, eb->key);   S32 result = cValue.getInt();   S32 res = result < 0 ? -1 : ( result > 0 ? 1 : 0 );   return ( smDecreasing ? -res : res );}S32 QSORT_CALLBACK ArrayObject::_valueFunctionCompare( const void* a, const void* b ){   ArrayObject::Element* ea = ( ArrayObject::Element* )( a );   ArrayObject::Element* eb = ( ArrayObject::Element* )( b );      ConsoleValue cValue = Con::executef( (const char*)smCompareFunction, ea->value, eb->value );   S32 result = cValue.getInt();   S32 res = result < 0 ? -1 : ( result > 0 ? 1 : 0 );   return ( smDecreasing ? -res : res );}//-----------------------------------------------------------------------------ArrayObject::ArrayObject()   : mCaseSensitive( false ),     mCurrentIndex( 0 ){}//-----------------------------------------------------------------------------void ArrayObject::initPersistFields(){   addField( "caseSensitive",    TypeBool,   Offset( mCaseSensitive, ArrayObject ),       "Makes the keys and values case-sensitive.\n"      "By default, comparison of key and value strings will be case-insensitive." );   addProtectedField( "key", TypeCaseString, 0, &_addKeyFromField, &emptyStringProtectedGetFn,      "Helper field which allows you to add new key['keyname'] = value pairs." );   Parent::initPersistFields();}//-----------------------------------------------------------------------------bool ArrayObject::_addKeyFromField( void *object, const char *index, const char *data ){   static_cast<ArrayObject*>( object )->push_back( index, data );   return false;}//-----------------------------------------------------------------------------S32 ArrayObject::getIndexFromValue( const String &value ) const{   S32 foundIndex = -1;   for ( S32 i = mCurrentIndex; i < mArray.size(); i++ )   {      if ( isEqual( mArray[i].value, value ) )      {         foundIndex = i;         break;      }   }   if( foundIndex < 0 )   {      for ( S32 i = 0; i < mCurrentIndex; i++ )      {         if ( isEqual( mArray[i].value, value ) )         {            foundIndex = i;            break;         }      }   }   return foundIndex;}//-----------------------------------------------------------------------------S32 ArrayObject::getIndexFromKey( const String &key ) const{   S32 foundIndex = -1;   for ( S32 i = mCurrentIndex; i < mArray.size(); i++ )   {      if ( isEqual( mArray[i].key, key ) )      {         foundIndex = i;         break;      }   }   if( foundIndex < 0 )   {      for ( S32 i = 0; i < mCurrentIndex; i++ )      {         if ( isEqual( mArray[i].key, key ) )         {            foundIndex = i;            break;         }      }   }   return foundIndex;}//-----------------------------------------------------------------------------S32 ArrayObject::getIndexFromKeyValue( const String &key, const String &value ) const{   S32 foundIndex = -1;   for ( S32 i = mCurrentIndex; i < mArray.size(); i++ )   {      if ( isEqual( mArray[i].key, key ) && isEqual( mArray[i].value, value ) )      {         foundIndex = i;         break;      }   }   if ( foundIndex < 0 )   {      for ( S32 i = 0; i < mCurrentIndex; i++ )      {         if ( isEqual( mArray[i].key, key ) && isEqual( mArray[i].value, value ) )         {            foundIndex = i;            break;         }      }   }   return foundIndex;}//-----------------------------------------------------------------------------const String& ArrayObject::getKeyFromIndex( S32 index ) const{   if ( index >= mArray.size() || index < 0 )      return String::EmptyString;   return mArray[index].key;}//-----------------------------------------------------------------------------const String& ArrayObject::getValueFromIndex( S32 index ) const{   if( index >= mArray.size() || index < 0 )      return String::EmptyString;   return mArray[index].value;}//-----------------------------------------------------------------------------S32 ArrayObject::countValue( const String &value ) const{   S32 count = 0;   for ( S32 i = 0; i < mArray.size(); i++ )   {      if ( isEqual( mArray[i].value, value ) )         count++;   }   return count;}//-----------------------------------------------------------------------------S32 ArrayObject::countKey( const String &key) const {   S32 count = 0;   for ( S32 i = 0; i < mArray.size(); i++ )   {      if ( isEqual( mArray[i].key, key ) )         count++;   }   return count;}//-----------------------------------------------------------------------------void ArrayObject::push_back( const String &key, const String &value ){   mArray.push_back( Element( key, value ) );}//-----------------------------------------------------------------------------void ArrayObject::push_front( const String &key, const String &value ){   mArray.push_front( Element( key, value ) );}//-----------------------------------------------------------------------------void ArrayObject::insert( const String &key, const String &value, S32 index ){   index = mClamp( index, 0, mArray.size() );   mArray.insert( index, Element( key, value ) );}//-----------------------------------------------------------------------------void ArrayObject::pop_back(){   if(mArray.size() <= 0)      return;   mArray.pop_back();   if( mCurrentIndex >= mArray.size() )      mCurrentIndex = mArray.size() - 1;}//-----------------------------------------------------------------------------void ArrayObject::pop_front(){   if( mArray.size() <= 0 )      return;   mArray.pop_front();      if( mCurrentIndex >= mArray.size() )      mCurrentIndex = mArray.size() - 1;}//-----------------------------------------------------------------------------void ArrayObject::erase( S32 index ){   if(index < 0 || index >= mArray.size())      return;   mArray.erase( index );}//-----------------------------------------------------------------------------void ArrayObject::empty(){   mArray.clear();   mCurrentIndex = 0;}//-----------------------------------------------------------------------------void ArrayObject::moveIndex(S32 prev, S32 index){   if(index >= mArray.size())      push_back(mArray[prev].key, mArray[prev].value);   else      mArray[index] = mArray[prev];   mArray[prev].value = String::EmptyString;   mArray[prev].key = String::EmptyString;}//-----------------------------------------------------------------------------void ArrayObject::uniqueValue(){   for(S32 i=0; i<mArray.size(); i++)   {      for(S32 j=i+1; j<mArray.size(); j++)      {         if ( isEqual( mArray[i].value, mArray[j].value ) )         {            erase(j);            j--;         }      }   }}//-----------------------------------------------------------------------------void ArrayObject::uniqueKey(){   for(S32 i=0; i<mArray.size(); i++)   {      for(S32 j=i+1; j<mArray.size(); j++)      {         if( isEqual( mArray[i].key, mArray[j].key ) )         {            erase(j);            j--;         }      }   }}//-----------------------------------------------------------------------------void ArrayObject::duplicate(ArrayObject* obj){   empty();   for(S32 i=0; i<obj->count(); i++)   {      const String& tempval = obj->getValueFromIndex(i);      const String& tempkey = obj->getKeyFromIndex(i);      push_back(tempkey, tempval);   }   mCurrentIndex = obj->getCurrent();}//-----------------------------------------------------------------------------void ArrayObject::crop( ArrayObject *obj ){   for( S32 i = 0; i < obj->count(); i++ )   {      const String &tempkey = obj->getKeyFromIndex( i );      for( S32 j = 0; j < mArray.size(); j++ )      {         if( isEqual( mArray[j].key, tempkey ) )         {            mArray.erase( j );            j--;         }      }   }}//-----------------------------------------------------------------------------void ArrayObject::append(ArrayObject* obj){   for(S32 i=0; i<obj->count(); i++)   {      const String& tempval = obj->getValueFromIndex(i);      const String& tempkey = obj->getKeyFromIndex(i);      push_back(tempkey, tempval);   }}//-----------------------------------------------------------------------------void ArrayObject::setKey( const String &key, S32 index ){   if ( index >= mArray.size() )      return;   mArray[index].key = key;}//-----------------------------------------------------------------------------void ArrayObject::setValue( const String &value, S32 index ){   if ( index >= mArray.size() )      return;      mArray[index].value = value;}//-----------------------------------------------------------------------------void ArrayObject::sort( bool valsort, bool asc, bool numeric ){   if ( mArray.size() <= 1 )      return;   smDecreasing = asc ? false : true;   smCaseSensitive = isCaseSensitive();   if ( numeric )   {      if ( valsort )         dQsort( (void *)&(mArray[0]), mArray.size(), sizeof(Element), _valueNumCompare) ;      else         dQsort( (void *)&(mArray[0]), mArray.size(), sizeof(Element), _keyNumCompare );   }   else   {      if( valsort )         dQsort( (void *)&(mArray[0]), mArray.size(), sizeof(Element), _valueCompare );      else         dQsort( (void *)&(mArray[0]), mArray.size(), sizeof(Element), _keyCompare );   }}//-----------------------------------------------------------------------------void ArrayObject::sort( bool valsort, bool asc, const char* callbackFunctionName ){   if( mArray.size() <= 1 )      return;   smDecreasing = asc ? false : true;   smCompareFunction = callbackFunctionName;   if( valsort )      dQsort( ( void* ) &( mArray[ 0 ] ), mArray.size(), sizeof( Element ), _valueFunctionCompare ) ;   else      dQsort( ( void* ) &( mArray[ 0 ] ), mArray.size(), sizeof( Element ), _keyFunctionCompare );   smCompareFunction = NULL;}//-----------------------------------------------------------------------------S32 ArrayObject::moveFirst(){   mCurrentIndex = 0;   return mCurrentIndex;}//-----------------------------------------------------------------------------S32 ArrayObject::moveLast(){   if ( mArray.empty() )      mCurrentIndex = 0;   else      mCurrentIndex = mArray.size() - 1;   return mCurrentIndex;}//-----------------------------------------------------------------------------S32 ArrayObject::moveNext(){   if ( mCurrentIndex >= mArray.size() - 1 )      return -1;      mCurrentIndex++;      return mCurrentIndex;}//-----------------------------------------------------------------------------S32 ArrayObject::movePrev(){   if ( mCurrentIndex <= 0 )      return -1;   mCurrentIndex--;      return mCurrentIndex;}//-----------------------------------------------------------------------------void ArrayObject::setCurrent( S32 idx ){   if ( idx < 0 || idx >= mArray.size() )   {      Con::errorf( "ArrayObject::setCurrent( %d ) is out of the array bounds!", idx );      return;   }   mCurrentIndex = idx;}//-----------------------------------------------------------------------------void ArrayObject::echo(){   Con::printf( "ArrayObject Listing:" );   Con::printf( "Index   Key       Value" );   for ( U32 i = 0; i < mArray.size(); i++ )   {      const String& key = mArray[i].key;      const String& val = mArray[i].value;      Con::printf( "%d      [%s]    =>    %s", i, key.c_str(), val.c_str() );   }}//=============================================================================//    Console Methods.//=============================================================================DefineEngineMethod( ArrayObject, getIndexFromValue, S32, ( const char* value ),,   "Search the array from the current position for the element "   "@param value Array value to search for\n"   "@return Index of the first element found, or -1 if none\n" ){   return object->getIndexFromValue( value );}DefineEngineMethod( ArrayObject, getIndexFromKey, S32, ( const char* key ),,   "Search the array from the current position for the key "   "@param value Array key to search for\n"   "@return Index of the first element found, or -1 if none\n" ){   return object->getIndexFromKey( key );}DefineEngineMethod( ArrayObject, getValue, const char*, ( S32 index ),,   "Get the value of the array element at the submitted index.\n"   "@param index 0-based index of the array element to get\n"   "@return The value of the array element at the specified index, "   "or \"\" if the index is out of range\n" ){   return object->getValueFromIndex( index ).c_str();}DefineEngineMethod( ArrayObject, getKey, const char*, ( S32 index ),,   "Get the key of the array element at the submitted index.\n"   "@param index 0-based index of the array element to get\n"   "@return The key associated with the array element at the "   "specified index, or \"\" if the index is out of range\n" ){   return object->getKeyFromIndex( index ).c_str();}DefineEngineMethod( ArrayObject, setKey, void, ( const char* key, S32 index ),,   "Set the key at the given index.\n"   "@param key New key value\n"   "@param index 0-based index of the array element to update\n" ){   object->setKey( key, index );}DefineEngineMethod( ArrayObject, setValue, void, ( const char* value, S32 index ),,   "Set the value at the given index.\n"   "@param value New array element value\n"   "@param index 0-based index of the array element to update\n" ){   object->setValue( value, index );}DefineEngineMethod( ArrayObject, count, S32, (),,   "Get the number of elements in the array." ){   return (S32)object->count();}DefineEngineMethod( ArrayObject, countValue, S32, ( const char* value ),,   "Get the number of times a particular value is found in the array.\n"   "@param value Array element value to count\n" ){   return (S32)object->countValue( value );}DefineEngineMethod( ArrayObject, countKey, S32, ( const char* key ),,   "Get the number of times a particular key is found in the array.\n"   "@param key Key value to count\n" ){   return (S32)object->countKey( key );}DefineEngineMethod( ArrayObject, add, void, ( const char* key, const char* value ), ( "" ),   "Adds a new element to the end of an array (same as push_back()).\n"   "@param key Key for the new element\n"   "@param value Value for the new element\n" ){   object->push_back( key, value );}DefineEngineMethod( ArrayObject, push_back, void, ( const char* key, const char* value ), ( "" ),   "Adds a new element to the end of an array.\n"   "@param key Key for the new element\n"   "@param value Value for the new element\n" ){   object->push_back( key, value );}DefineEngineMethod( ArrayObject, push_front, void, ( const char* key, const char* value ), ( "" ),   "Adds a new element to the front of an array" ){   object->push_front( key, value );}DefineEngineMethod( ArrayObject, insert, void, ( const char* key, const char* value, S32 index ),,   "Adds a new element to a specified position in the array.\n"   "- @a index = 0 will insert an element at the start of the array (same as push_front())\n"   "- @a index = %array.count() will insert an element at the end of the array (same as push_back())\n\n"   "@param key Key for the new element\n"   "@param value Value for the new element\n"   "@param index 0-based index at which to insert the new element" ){   object->insert( key, value, index );}DefineEngineMethod( ArrayObject, pop_back, void, (),,   "Removes the last element from the array" ){   object->pop_back();}DefineEngineMethod( ArrayObject, pop_front, void, (),,   "Removes the first element from the array" ){   object->pop_front();}DefineEngineMethod( ArrayObject, erase, void, ( S32 index ),,   "Removes an element at a specific position from the array.\n"   "@param index 0-based index of the element to remove\n" ){   object->erase( index );}DefineEngineMethod( ArrayObject, empty, void, (),,   "Emptys all elements from an array" ){   object->empty();}DefineEngineMethod( ArrayObject, uniqueValue, void, (),,   "Removes any elements that have duplicated values (leaving the first instance)" ){   object->uniqueValue();}DefineEngineMethod( ArrayObject, uniqueKey, void, (),,   "Removes any elements that have duplicated keys (leaving the first instance)" ){   object->uniqueKey();}DefineEngineMethod( ArrayObject, duplicate, bool, ( ArrayObject* target ),,   "Alters array into an exact duplicate of the target array.\n"   "@param target ArrayObject to duplicate\n" ){   if ( target )   {      object->duplicate( target );      return true;   }   return false;}DefineEngineMethod( ArrayObject, crop, bool, ( ArrayObject* target ),,   "Removes elements with matching keys from array.\n"   "@param target ArrayObject containing keys to remove from this array\n" ){   if ( target )   {      object->crop( target );      return true;   }   return false;}DefineEngineMethod( ArrayObject, append, bool, ( ArrayObject* target ),,   "Appends the target array to the array object.\n"   "@param target ArrayObject to append to the end of this array\n" ){   if ( target )   {      object->append( target );      return true;   }   return false;}DefineEngineMethod( ArrayObject, sort, void, ( bool ascending ), ( false ),   "Alpha sorts the array by value\n\n"   "@param ascending [optional] True for ascending sort, false for descending sort\n" ){   object->sort( true, ascending, false );}DefineEngineMethod( ArrayObject, sorta, void, (),,   "Alpha sorts the array by value in ascending order" ){   object->sort( true, true, false );}DefineEngineMethod( ArrayObject, sortd, void, (),,   "Alpha sorts the array by value in descending order" ){   object->sort( true, false, false );}DefineEngineMethod( ArrayObject, sortk, void, ( bool ascending ), ( false ),   "Alpha sorts the array by key\n\n"   "@param ascending [optional] True for ascending sort, false for descending sort\n" ){   object->sort( false, ascending, false );}DefineEngineMethod( ArrayObject, sortka, void, (),,   "Alpha sorts the array by key in ascending order" ){   object->sort( false, true, false );}DefineEngineMethod( ArrayObject, sortkd, void, (),,   "Alpha sorts the array by key in descending order" ){   object->sort( false, false, false );}DefineEngineMethod( ArrayObject, sortn, void, ( bool ascending ), ( false ),   "Numerically sorts the array by value\n\n"   "@param ascending [optional] True for ascending sort, false for descending sort\n" ){   object->sort( true, ascending, true );}DefineEngineMethod( ArrayObject, sortna, void, (),,   "Numerically sorts the array by value in ascending order" ) {   object->sort( true, true, true );}DefineEngineMethod( ArrayObject, sortnd, void, (),,   "Numerically sorts the array by value in descending order" ){   object->sort( true, false, true );}DefineEngineMethod( ArrayObject, sortnk, void, ( bool ascending ), ( false ),   "Numerically sorts the array by key\n\n"   "@param ascending [optional] True for ascending sort, false for descending sort\n" ){   object->sort( false, ascending, true );}DefineEngineMethod( ArrayObject, sortnka, void, (),,   "Numerical sorts the array by key in ascending order" ){   object->sort( false, true, true );}DefineEngineMethod( ArrayObject, sortnkd, void, (),,   "Numerical sorts the array by key in descending order" ){   object->sort( false, false, true );}DefineEngineMethod( ArrayObject, sortf, void,  ( const char* functionName ),,   "Sorts the array by value in ascending order using the given callback function.\n"   "@param functionName Name of a function that takes two arguments A and B and returns -1 if A is less, 1 if B is less, and 0 if both are equal.\n\n"   "@tsexample\n"   "function mySortCallback(%a, %b)\n"   "{\n"   "   return strcmp( %a.name, %b.name );\n"   "}\n\n"   "%array.sortf( \"mySortCallback\" );\n"   "@endtsexample\n" ){   object->sort( true, true, functionName );}DefineEngineMethod( ArrayObject, sortfk, void,  ( const char* functionName ),,   "Sorts the array by key in ascending order using the given callback function.\n"   "@param functionName Name of a function that takes two arguments A and B and returns -1 if A is less, 1 if B is less, and 0 if both are equal."   "@see sortf\n" ){   object->sort( false, true, functionName );}DefineEngineMethod( ArrayObject, sortfd, void, ( const char* functionName ),,   "Sorts the array by value in descending order using the given callback function.\n"   "@param functionName Name of a function that takes two arguments A and B and returns -1 if A is less, 1 if B is less, and 0 if both are equal."   "@see sortf\n" ){   object->sort( true, false, functionName );}DefineEngineMethod( ArrayObject, sortfkd, void, ( const char* functionName ),,   "Sorts the array by key in descending order using the given callback function.\n"   "@param functionName Name of a function that takes two arguments A and B and returns -1 if A is less, 1 if B is less, and 0 if both are equal."   "@see sortf\n" ){   object->sort( false, false, functionName );}DefineEngineMethod( ArrayObject, moveFirst, S32, (),,   "Moves array pointer to start of array\n\n"   "@return Returns the new array pointer" ){   return object->moveFirst();}DefineEngineMethod( ArrayObject, moveLast, S32, (),,   "Moves array pointer to end of array\n\n"   "@return Returns the new array pointer" ){   return object->moveLast();}DefineEngineMethod( ArrayObject, moveNext, S32, (),,   "Moves array pointer to next position\n\n"   "@return Returns the new array pointer, or -1 if already at the end" ){   return object->moveNext();}DefineEngineMethod( ArrayObject, movePrev, S32, (),,   "Moves array pointer to prev position\n\n"   "@return Returns the new array pointer, or -1 if already at the start" ){   return object->movePrev();}DefineEngineMethod( ArrayObject, getCurrent, S32, (),,   "Gets the current pointer index" ){   return object->getCurrent();}DefineEngineMethod( ArrayObject, setCurrent, void, ( S32 index ),,   "Sets the current pointer index.\n"   "@param index New 0-based pointer index\n" ){   object->setCurrent( index );}DefineEngineMethod( ArrayObject, echo, void, (),,   "Echos the array contents to the console" ){   object->echo();}
 |