| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989 | //-----------------------------------------------------------------------------// 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 "gui/worldEditor/terrainEditor.h"#include "core/frameAllocator.h"#include "core/strings/stringUnit.h"#include "console/consoleTypes.h"#include "console/simEvents.h"#include "console/engineAPI.h"#include "sim/netConnection.h"#include "math/mathUtils.h"#include "gfx/primBuilder.h"#include "gfx/gfxDrawUtil.h"#include "gui/core/guiCanvas.h"#include "gui/worldEditor/terrainActions.h"#include "terrain/terrMaterial.h"#include "T3D/Scene.h"IMPLEMENT_CONOBJECT(TerrainEditor);ConsoleDocClass( TerrainEditor,   "@brief The base Terrain Editor tool\n\n"   "Editor use only.\n\n"   "@internal");Selection::Selection() :   Vector<GridInfo>(__FILE__, __LINE__),   mName(0),   mUndoFlags(0),   mHashListSize(1024){   VECTOR_SET_ASSOCIATION(mHashLists);   // clear the hash list   mHashLists.setSize(mHashListSize);   reset();}Selection::~Selection(){}void Selection::reset(){   PROFILE_SCOPE( TerrainEditor_Selection_Reset );   for(U32 i = 0; i < mHashListSize; i++)      mHashLists[i] = -1;   clear();}bool Selection::validate(){   PROFILE_SCOPE( TerrainEditor_Selection_Validate );   // scan all the hashes and verify that the heads they point to point back to them   U32 hashesProcessed = 0;   for(U32 i = 0; i < mHashLists.size(); i++)   {      U32 entry = mHashLists[i];      if(entry == -1)         continue;      GridInfo info = (*this)[entry];      U32 hashIndex = getHashIndex(info.mGridPoint.gridPos);      if( entry != mHashLists[hashIndex] )      {         AssertFatal(false, "Selection hash lists corrupted");         return false;      }      hashesProcessed++;   }   // scan all the entries and verify that anything w/ a prev == -1 is correctly in the hash table   U32 headsProcessed = 0;   for(U32 i = 0; i < size(); i++)   {      GridInfo info = (*this)[i];      if(info.mPrev != -1)         continue;      U32 hashIndex = getHashIndex(info.mGridPoint.gridPos);      if(mHashLists[hashIndex] != i)      {         AssertFatal(false, "Selection list heads corrupted");         return false;      }      headsProcessed++;   }   AssertFatal(headsProcessed == hashesProcessed, "Selection's number of hashes and number of list heads differ.");   return true;}U32 Selection::getHashIndex(const Point2I & pos){   PROFILE_SCOPE( TerrainEditor_Selection_GetHashIndex );   Point2F pnt = Point2F((F32)pos.x, (F32)pos.y) + Point2F(1.3f,3.5f);   return( (U32)(mFloor(mHashLists.size() * mFmod(pnt.len() * 0.618f, 1.0f))) );}S32 Selection::lookup(const Point2I & pos){   PROFILE_SCOPE( TerrainEditor_Selection_Lookup );   U32 index = getHashIndex(pos);   S32 entry = mHashLists[index];   while(entry != -1)   {      if((*this)[entry].mGridPoint.gridPos == pos)         return(entry);      entry = (*this)[entry].mNext;   }   return(-1);}void Selection::insert(GridInfo info){   PROFILE_SCOPE( TerrainEditor_Selection_Insert );   //validate();   // get the index into the hash table   U32 index = getHashIndex(info.mGridPoint.gridPos);   // if there is an existing linked list, make it our next   info.mNext = mHashLists[index];   info.mPrev = -1;   // if there is an existing linked list, make us it's prev   U32 indexOfNewEntry = size();   if(info.mNext != -1)      (*this)[info.mNext].mPrev = indexOfNewEntry;   // the hash table holds the heads of the linked lists. make us the head of this list.   mHashLists[index] = indexOfNewEntry;   // copy us into the vector   push_back(info);   //validate();}bool Selection::remove(const GridInfo &info){   PROFILE_SCOPE( TerrainEditor_Selection_Remove );   if(size() < 1)      return false;   //AssertFatal( validate(), "Selection hashLists corrupted before Selection.remove()");   U32 hashIndex = getHashIndex(info.mGridPoint.gridPos);   S32 listHead = mHashLists[hashIndex];   //AssertFatal(listHead < size(), "A Selection's hash table is corrupt.");   if(listHead == -1)      return(false);   const S32 victimEntry = lookup(info.mGridPoint.gridPos);   if( victimEntry == -1 )      return(false);   const GridInfo victim = (*this)[victimEntry];   const S32 vicPrev = victim.mPrev;   const S32 vicNext = victim.mNext;   // remove us from the linked list, if there is one.   if(vicPrev != -1)      (*this)[vicPrev].mNext = vicNext;   if(vicNext != -1)      (*this)[vicNext].mPrev = vicPrev;   // if we were the head of the list, make our next the new head in the hash table.   if(vicPrev == -1)      mHashLists[hashIndex] = vicNext;   // if we're not the last element in the vector, copy the last element to our position.   if(victimEntry != size() - 1)   {      // copy last into victim, and re-cache next & prev      const GridInfo lastEntry = last();      const S32 lastPrev = lastEntry.mPrev;      const S32 lastNext = lastEntry.mNext;      (*this)[victimEntry] = lastEntry;      // update the new element's next and prev, to reestablish it in it's linked list.      if(lastPrev != -1)         (*this)[lastPrev].mNext = victimEntry;      if(lastNext != -1)         (*this)[lastNext].mPrev = victimEntry;      // if it was the head of it's list, update the hash table with its new position.      if(lastPrev == -1)      {         const U32 lastHash = getHashIndex(lastEntry.mGridPoint.gridPos);         AssertFatal(mHashLists[lastHash] == size() - 1, "Selection hashLists corrupted during Selection.remove() (oldmsg)");         mHashLists[lastHash] = victimEntry;      }   }   // decrement the vector, we're done here   pop_back();   //AssertFatal( validate(), "Selection hashLists corrupted after Selection.remove()");   return true;}bool Selection::add(const GridInfo &info){   PROFILE_SCOPE( TerrainEditor_Selection_Add );   S32 index = lookup(info.mGridPoint.gridPos);   if(index != -1)      return(false);   insert(info);   return(true);}bool Selection::getInfo(Point2I pos, GridInfo & info){   PROFILE_SCOPE( TerrainEditor_Selection_GetInfo );   S32 index = lookup(pos);   if(index == -1)      return(false);   info = (*this)[index];   return(true);}bool Selection::setInfo(GridInfo & info){   PROFILE_SCOPE( TerrainEditor_Selection_SetInfo );   S32 index = lookup(info.mGridPoint.gridPos);   if(index == -1)      return(false);   S32 next = (*this)[index].mNext;   S32 prev = (*this)[index].mPrev;   (*this)[index] = info;   (*this)[index].mNext = next;   (*this)[index].mPrev = prev;   return(true);}F32 Selection::getAvgHeight(){   PROFILE_SCOPE( TerrainEditor_Selection_GetAvgHeight );   if(!size())      return(0);   F32 avg = 0.f;   for(U32 i = 0; i < size(); i++)      avg += (*this)[i].mHeight;   return(avg / size());}F32 Selection::getMinHeight(){   PROFILE_SCOPE( TerrainEditor_Selection_GetMinHeight );   if(!size())      return(0);   F32 minHeight = (*this)[0].mHeight;   for(U32 i = 1; i < size(); i++)      minHeight = getMin(minHeight, (*this)[i].mHeight);   return minHeight;}F32 Selection::getMaxHeight(){   PROFILE_SCOPE( TerrainEditor_Selection_GetMaxHeight );   if(!size())      return(0);   F32 maxHeight = (*this)[0].mHeight;   for(U32 i = 1; i < size(); i++)      maxHeight = getMax(maxHeight, (*this)[i].mHeight);   return maxHeight;}Brush::Brush(TerrainEditor * editor) :   mTerrainEditor(editor){   mSize = mTerrainEditor->getBrushSize();}const Point2I & Brush::getPosition(){   return(mGridPoint.gridPos);}const GridPoint & Brush::getGridPoint(){   return mGridPoint;}void Brush::setPosition(const Point3F & pos){   PROFILE_SCOPE( TerrainEditor_Brush_SetPosition_Point3F );   mTerrainEditor->worldToGrid(pos, mGridPoint);   update();}void Brush::setPosition(const Point2I & pos){   PROFILE_SCOPE( TerrainEditor_Brush_SetPosition_Point2I );   mGridPoint.gridPos = pos;   update();}void Brush::update(){   PROFILE_SCOPE( TerrainEditor_Brush_update );   if ( mGridPoint.terrainBlock )      rebuild();}void Brush::render(){   PROFILE_SCOPE( TerrainEditor_Brush_Render );   // Render the brush's outline via the derived brush class.   _renderOutline();   // Render the brush's interior grid points.   const U32 pointCount = mSize.x * mSize.y;   if ( pointCount == 0 )      return;   if ( mRenderList.empty() || empty() )      return;   Vector<GFXVertexPCT> pointList;   pointList.reserve( pointCount );   for(S32 x = 0; x < mSize.x; x++)   {      for(S32 y = 0; y < mSize.y; y++)      {         S32 id = mRenderList[x*mSize.x+y];         if ( id == -1 )            continue;         const GridInfo &gInfo = (*this)[ id ];         Point3F pos;         mTerrainEditor->gridToWorld( gInfo.mGridPoint.gridPos, pos, gInfo.mGridPoint.terrainBlock );         if ( !mTerrainEditor->project( pos, &pos ) )            continue;         pointList.increment();         GFXVertexPCT &pointInfo = pointList.last();         pointInfo.point = pos;         pointInfo.color.set( 255, 0, 255, gInfo.mWeight * 255 );         pointInfo.texCoord.set( 1.0f, 0.0f );      }   }   mTerrainEditor->renderPoints( pointList );}void BoxBrush::rebuild(){   PROFILE_SCOPE( TerrainEditor_BoxBrush_Rebuild );   reset();   const F32 squareSize = mGridPoint.terrainBlock->getSquareSize();   mRenderList.setSize(mSize.x*mSize.y);   Point3F center( F32(mSize.x - 1) / 2.0f * squareSize, F32(mSize.y - 1) / 2.0f * squareSize, 0.0f );   Filter filter;   filter.set(1, &mTerrainEditor->mSoftSelectFilter);   const Point3F mousePos = mTerrainEditor->getMousePos();   F32 xFactorScale = center.x / ( center.x + 0.5f );   F32 yFactorScale = center.y / ( center.y + 0.5f );   const F32 softness = mTerrainEditor->getBrushSoftness();   const F32 pressure = mTerrainEditor->getBrushPressure();   Point3F posw( 0,0,0 );   Point2I posg( 0,0 );   Vector<GridInfo> infos;   for ( S32 x = 0; x < mSize.x; x++ )   {      for(S32 y = 0; y < mSize.y; y++)      {         F32 xFactor = 0.0f;         if ( center.x > 0 )            xFactor = mAbs( center.x - x ) / center.x * xFactorScale;         F32 yFactor = 0.0f;         if ( center.y > 0 )            yFactor = mAbs( center.y - y ) / center.y * yFactorScale;         S32 &rl = mRenderList[x*mSize.x+y];         posw.x = mousePos.x + (F32)x * squareSize - center.x;         posw.y = mousePos.y + (F32)y * squareSize - center.y;         // round to grid coords         GridPoint gridPoint = mGridPoint;         mTerrainEditor->worldToGrid( posw, gridPoint );         // Check that the grid point is valid within the terrain.  This assumes         // that there is no wrap around past the edge of the terrain.         if(!mTerrainEditor->isPointInTerrain(gridPoint))         {            rl = -1;            continue;         }         infos.clear();         mTerrainEditor->getGridInfos( gridPoint, infos );         for (U32 z = 0; z < infos.size(); z++)         {            infos[z].mWeight = pressure *               mLerp( infos[z].mWeight, filter.getValue(xFactor > yFactor ? xFactor : yFactor), softness );            push_back(infos[z]);         }         rl = size()-1;      }   }}void BoxBrush::_renderOutline(){   F32 squareSize = mGridPoint.terrainBlock->getSquareSize();   RayInfo ri;   Point3F start( 0, 0, 5000.0f );   Point3F end( 0, 0, -5000.0f );   bool hit;   Vector<Point3F> pointList;   pointList.reserve( 64 );   const ColorI col( 255, 0, 255, 255 );   const Point3F &mousePos = mTerrainEditor->getMousePos();   static const Point2F offsetArray [5] =   {      Point2F( -1, -1 ),      Point2F( 1, -1 ),      Point2F( 1, 1 ),      Point2F( -1, 1 ),      Point2F( -1, -1 ) // repeat of offset[0]   };   // 64 total steps, 4 sides to the box, 16 steps per side.   // 64 / 4 = 16   const U32 steps = 16;   for ( S32 i = 0; i < 4; i++ )   {      const Point2F &offset = offsetArray[i];      const Point2F &next = offsetArray[i+1];      for ( S32 j = 0; j < steps; j++ )      {         F32 frac = (F32)j / ( (F32)steps - 1.0f );         Point2F tmp;         tmp.interpolate( offset, next, frac );         start.x = end.x = mousePos.x + tmp.x * squareSize * 0.5f * (F32)mSize.x;         start.y = end.y = mousePos.y + tmp.y * squareSize * 0.5f * (F32)mSize.y;         hit = gServerContainer.castRay( start, end, TerrainObjectType, &ri );         if ( hit )            pointList.push_back( ri.point );      }   }   mTerrainEditor->drawLineList( pointList, col, 1.0f );}void EllipseBrush::rebuild(){   PROFILE_SCOPE( TerrainEditor_EllipseBrush_Rebuild );   reset();   const F32 squareSize = mGridPoint.terrainBlock->getSquareSize();   mRenderList.setSize(mSize.x*mSize.y);   Point3F center( F32(mSize.x - 1) / 2.0f * squareSize, F32(mSize.y - 1) / 2.0f * squareSize, 0.0f );   Filter filter;   filter.set(1, &mTerrainEditor->mSoftSelectFilter);   const Point3F mousePos = mTerrainEditor->getMousePos();   // a point is in a circle if:   // x^2 + y^2 <= r^2   // a point is in an ellipse if:   // (ax)^2 + (by)^2 <= 1   // where a = 1/halfEllipseWidth and b = 1/halfEllipseHeight   // for a soft-selected ellipse,   // the factor is simply the filtered: ((ax)^2 + (by)^2)   F32 a = 1.0f / (F32(mSize.x) * squareSize * 0.5f);   F32 b = 1.0f / (F32(mSize.y) * squareSize * 0.5f);   const F32 softness = mTerrainEditor->getBrushSoftness();   const F32 pressure = mTerrainEditor->getBrushPressure();   Point3F posw( 0,0,0 );   Point2I posg( 0,0 );   Vector<GridInfo> infos;   for ( S32 x = 0; x < mSize.x; x++ )   {      for ( S32 y = 0; y < mSize.y; y++ )      {         F32 xp = center.x - x * squareSize;         F32 yp = center.y - y * squareSize;         F32 factor = (a * a * xp * xp) + (b * b * yp * yp);         if ( factor > 1 )         {            mRenderList[x*mSize.x+y] = -1;            continue;         }         S32 &rl = mRenderList[x*mSize.x+y];         posw.x = mousePos.x + (F32)x * squareSize - center.x;         posw.y = mousePos.y + (F32)y * squareSize - center.y;         // round to grid coords         GridPoint gridPoint = mGridPoint;         mTerrainEditor->worldToGrid( posw, gridPoint );         // Check that the grid point is valid within the terrain.  This assumes         // that there is no wrap around past the edge of the terrain.         if ( !mTerrainEditor->isPointInTerrain( gridPoint ) )         {            rl = -1;            continue;         }         infos.clear();         mTerrainEditor->getGridInfos( gridPoint, infos );         for ( U32 z = 0; z < infos.size(); z++ )         {            infos[z].mWeight = pressure * mLerp( infos[z].mWeight, filter.getValue( factor ), softness );            push_back(infos[z]);         }         rl = size()-1;      }   }}void EllipseBrush::_renderOutline(){   F32 squareSize = mGridPoint.terrainBlock->getSquareSize();   RayInfo ri;   Point3F start( 0, 0, 5000.0f );   Point3F end( 0, 0, -5000.0f );   bool hit;   Vector<Point3F> pointList;   ColorI col( 255, 0, 255, 255 );   const U32 steps = 64;   const Point3F &mousePos = mTerrainEditor->getMousePos();   for ( S32 i = 0; i < steps; i++ )   {      F32 radians = (F32)i / (F32)(steps-1) * M_2PI_F;      VectorF vec(0,1,0);      MathUtils::vectorRotateZAxis( vec, radians );      start.x = end.x = mousePos.x + vec.x * squareSize * (F32)mSize.x * 0.5f;      start.y = end.y = mousePos.y + vec.y * squareSize * (F32)mSize.y * 0.5f;      hit = gServerContainer.castRay( start, end, TerrainObjectType, &ri );      if ( hit )         pointList.push_back( ri.point );   }   mTerrainEditor->drawLineList( pointList, col, 1.0f );}SelectionBrush::SelectionBrush(TerrainEditor * editor) :   Brush(editor){   //... grab the current selection}void SelectionBrush::rebuild(){   reset();   //... move the selection}void SelectionBrush::render(Vector<GFXVertexPCT> & vertexBuffer, S32 & verts, S32 & elems, S32 & prims, const LinearColorF & inColorFull, const LinearColorF & inColorNone, const LinearColorF & outColorFull, const LinearColorF & outColorNone) const{   //... render the selection}TerrainEditor::TerrainEditor() :   mActiveTerrain(0),   mMousePos(0,0,0),   mMouseBrush(0),   mInAction(false),   mGridUpdateMin( S32_MAX, S32_MAX ),   mUndoSel(0),   mGridUpdateMax( 0, 0 ),   mNeedsGridUpdate( false ),   mMaxBrushSize(256,256),   mNeedsMaterialUpdate( false ),   mMouseDown( false ){   VECTOR_SET_ASSOCIATION(mActions);   //   resetCurrentSel();   //   mBrushPressure = 1.0f;   mBrushSize.set(1,1);   mBrushSoftness = 1.0f;   mBrushChanged = true;   mMouseBrush = new BoxBrush(this);   mMouseDownSeq = 0;   mIsDirty = false;   mIsMissionDirty = false;   mPaintIndex = -1;   // add in all the actions here..   mActions.push_back(new SelectAction(this));   mActions.push_back(new DeselectAction(this));   mActions.push_back(new ClearAction(this));   mActions.push_back(new SoftSelectAction(this));   mActions.push_back(new OutlineSelectAction(this));   mActions.push_back(new PaintMaterialAction(this));   mActions.push_back(new ClearMaterialsAction(this));   mActions.push_back(new RaiseHeightAction(this));   mActions.push_back(new LowerHeightAction(this));   mActions.push_back(new SetHeightAction(this));   mActions.push_back(new SetEmptyAction(this));   mActions.push_back(new ClearEmptyAction(this));   mActions.push_back(new ScaleHeightAction(this));   mActions.push_back(new BrushAdjustHeightAction(this));   mActions.push_back(new AdjustHeightAction(this));   mActions.push_back(new FlattenHeightAction(this));   mActions.push_back(new SmoothHeightAction(this));   mActions.push_back(new SmoothSlopeAction(this));   mActions.push_back(new PaintNoiseAction(this));   mActions.push_back(new ThermalErosionAction(this));   mActions.push_back(new HydraulicErosionAction(this));   mActions.push_back(new copyAction(this));   mActions.push_back(new pasteAction(this));   mActions.push_back(new pasteUpAction(this));   mActions.push_back(new pasteDownAction(this));      // set the default action   mCurrentAction = mActions[0];   mRenderBrush = mCurrentAction->useMouseBrush();   // persist data defaults   mRenderBorder = true;   mBorderHeight = 10;   mBorderFillColor.set(0,255,0,20);   mBorderFrameColor.set(0,255,0,128);   mBorderLineMode = false;   mSelectionHidden = false;   mRenderVertexSelection = false;   mRenderSolidBrush = false;   mProcessUsesBrush = false;   //   mAdjustHeightVal = 10;   mSetHeightVal = 100;   mScaleVal = 1;   mSmoothFactor = 0.1f;   mNoiseFactor = 1.0f;   mMaterialGroup = 0;   mSoftSelectRadius = 50.f;   mAdjustHeightMouseScale = 0.1f;   mSoftSelectDefaultFilter = StringTable->insert("1.000000 0.833333 0.666667 0.500000 0.333333 0.166667 0.000000");   mSoftSelectFilter = mSoftSelectDefaultFilter;   mSlopeMinAngle = 0.0f;   mSlopeMaxAngle = 90.0f;   mTileMinHeight = 0;   mTileMaxHeight = 2047;}TerrainEditor::~TerrainEditor(){   // mouse   delete mMouseBrush;   // terrain actions   U32 i;   for(i = 0; i < mActions.size(); i++)      delete mActions[i];   // undo stuff   delete mUndoSel;}TerrainAction * TerrainEditor::lookupAction(const char * name){   for(U32 i = 0; i < mActions.size(); i++)      if(!dStricmp(mActions[i]->getName(), name))         return(mActions[i]);   return(0);}bool TerrainEditor::onAdd(){   if ( !Parent::onAdd() )      return false;   GFXStateBlockDesc desc;   desc.setZReadWrite( false );   desc.zWriteEnable = false;   desc.setCullMode( GFXCullNone );   desc.setBlend( true, GFXBlendSrcAlpha, GFXBlendDestAlpha );   mStateBlock = GFX->createStateBlock( desc );   return true;}bool TerrainEditor::onWake(){   if ( !Parent::onWake() )      return false;   // Push our default cursor on here once.   GuiCanvas *root = getRoot();   if ( root )   {      S32 currCursor = PlatformCursorController::curArrow;      PlatformWindow *window = root->getPlatformWindow();      PlatformCursorController *controller = window->getCursorController();      controller->pushCursor( currCursor );   }   return true;}void TerrainEditor::onSleep(){   // Pop our default cursor off.   GuiCanvas *root = getRoot();   if ( root )   {      PlatformWindow *window = root->getPlatformWindow();      PlatformCursorController *controller = window->getCursorController();      controller->popCursor();   }   Parent::onSleep();}void TerrainEditor::get3DCursor( GuiCursor *&cursor,                                       bool &visible,                                       const Gui3DMouseEvent &event_ ){   cursor = NULL;   visible = false;   GuiCanvas *root = getRoot();   if ( !root )      return;   S32 currCursor = PlatformCursorController::curArrow;   if ( root->mCursorChanged == currCursor )      return;   PlatformWindow *window = root->getPlatformWindow();   PlatformCursorController *controller = window->getCursorController();   // We've already changed the cursor,   // so set it back before we change it again.   if( root->mCursorChanged != -1)      controller->popCursor();   // Now change the cursor shape   controller->pushCursor(currCursor);   root->mCursorChanged = currCursor;}void TerrainEditor::onDeleteNotify(SimObject * object){   Parent::onDeleteNotify(object);   if (dynamic_cast<TerrainBlock*>(object) == mActiveTerrain)      mActiveTerrain = NULL;}TerrainBlock* TerrainEditor::getClientTerrain( TerrainBlock *serverTerrain ) const{   if ( !serverTerrain )      serverTerrain = mActiveTerrain;   return serverTerrain ? dynamic_cast<TerrainBlock*>( serverTerrain->getClientObject() ) : NULL;}bool TerrainEditor::isMainTile(const GridPoint & gPoint) const{   const S32 blockSize = (S32)gPoint.terrainBlock->getBlockSize();   Point2I testPos = gPoint.gridPos;   if (!String::compare(getCurrentAction(),"paintMaterial"))   {      if (testPos.x == blockSize)         testPos.x--;      if (testPos.y == blockSize)         testPos.y--;   }   return (testPos.x >= 0 && testPos.x < blockSize && testPos.y >= 0 && testPos.y < blockSize);}TerrainBlock* TerrainEditor::getTerrainUnderWorldPoint(const Point3F & wPos){   PROFILE_SCOPE( TerrainEditor_GetTerrainUnderWorldPoint );   // Cast a ray straight down from the world position and see which   // Terrain is the closest to our starting point   Point3F startPnt = wPos;   Point3F endPnt = wPos + Point3F(0.0f, 0.0f, -1000.0f);   S32 blockIndex = -1;   F32 nearT = 1.0f;   for (U32 i = 0; i < mTerrainBlocks.size(); i++)   {      Point3F tStartPnt, tEndPnt;      mTerrainBlocks[i]->getWorldTransform().mulP(startPnt, &tStartPnt);      mTerrainBlocks[i]->getWorldTransform().mulP(endPnt, &tEndPnt);      RayInfo ri;      if (mTerrainBlocks[i]->castRayI(tStartPnt, tEndPnt, &ri, true))      {         if (ri.t < nearT)         {            blockIndex = i;            nearT = ri.t;         }      }   }   if (blockIndex > -1)      return mTerrainBlocks[blockIndex];   return NULL;}bool TerrainEditor::gridToWorld(const GridPoint & gPoint, Point3F & wPos){   PROFILE_SCOPE( TerrainEditor_GridToWorld );   const MatrixF & mat = gPoint.terrainBlock->getTransform();   Point3F origin;   mat.getColumn(3, &origin);   wPos.x = gPoint.gridPos.x * gPoint.terrainBlock->getSquareSize() + origin.x;   wPos.y = gPoint.gridPos.y * gPoint.terrainBlock->getSquareSize() + origin.y;   wPos.z = getGridHeight(gPoint) + origin.z;   return isMainTile(gPoint);}bool TerrainEditor::gridToWorld(const Point2I & gPos, Point3F & wPos, TerrainBlock* terrain){   GridPoint gridPoint;   gridPoint.gridPos = gPos;   gridPoint.terrainBlock = terrain;   return gridToWorld(gridPoint, wPos);}bool TerrainEditor::worldToGrid(const Point3F & wPos, GridPoint & gPoint){   PROFILE_SCOPE( TerrainEditor_WorldToGrid );   // If the grid point TerrainBlock is NULL then find the closest Terrain underneath that   // point - pad a little upward in case our incoming point already lies exactly on the terrain   if (!gPoint.terrainBlock)      gPoint.terrainBlock = getTerrainUnderWorldPoint(wPos + Point3F(0.0f, 0.0f, 0.05f));   if (gPoint.terrainBlock == NULL)      return false;   gPoint.gridPos = gPoint.terrainBlock->getGridPos(wPos);   return isMainTile(gPoint);}bool TerrainEditor::worldToGrid(const Point3F & wPos, Point2I & gPos, TerrainBlock* terrain){   GridPoint gridPoint;   gridPoint.terrainBlock = terrain;   bool ret = worldToGrid(wPos, gridPoint);   gPos = gridPoint.gridPos;   return ret;}bool TerrainEditor::gridToCenter(const Point2I & gPos, Point2I & cPos) const{   // TODO: What is this for... megaterrain or tiled terrains?   cPos.x = gPos.x; // & TerrainBlock::BlockMask;   cPos.y = gPos.y;// & TerrainBlock::BlockMask;   //if (gPos.x == TerrainBlock::BlockSize)   //   cPos.x = gPos.x;   //if (gPos.y == TerrainBlock::BlockSize)   //   cPos.y = gPos.y;   //return isMainTile(gPos);   return true;}//------------------------------------------------------------------------------//bool TerrainEditor::getGridInfo(const Point3F & wPos, GridInfo & info)//{//   Point2I gPos;//   worldToGrid(wPos, gPos);//   return getGridInfo(gPos, info);//}bool TerrainEditor::getGridInfo(const GridPoint & gPoint, GridInfo & info){   //   info.mGridPoint = gPoint;   info.mMaterial = getGridMaterial(gPoint);   info.mHeight = getGridHeight(gPoint);   info.mWeight = 1.f;   info.mPrimarySelect = true;   info.mMaterialChanged = false;   Point2I cPos;   gridToCenter(gPoint.gridPos, cPos);   return isMainTile(gPoint);}bool TerrainEditor::getGridInfo(const Point2I & gPos, GridInfo & info, TerrainBlock* terrain){   GridPoint gridPoint;   gridPoint.gridPos = gPos;   gridPoint.terrainBlock = terrain;   return getGridInfo(gridPoint, info);}void TerrainEditor::getGridInfos(const GridPoint & gPoint, Vector<GridInfo>& infos){   PROFILE_SCOPE( TerrainEditor_GetGridInfos );   // First we test against the brush terrain so that we can   // favor it (this should be the same as the active terrain)   bool foundBrush = false;   GridInfo baseInfo;   if (getGridInfo(gPoint, baseInfo))   {      infos.push_back(baseInfo);      foundBrush = true;   }   // We are going to need the world position to test against   Point3F wPos;   gridToWorld(gPoint, wPos);   // Now loop through our terrain blocks and decide which ones hit the point   // If we already found a hit against our brush terrain we only add points   // that are relatively close to the found point   for (U32 i = 0; i < mTerrainBlocks.size(); i++)   {      // Skip if we've already found the point on the brush terrain      if (foundBrush && mTerrainBlocks[i] == baseInfo.mGridPoint.terrainBlock)         continue;      // Get our grid position      Point2I gPos;      worldToGrid(wPos, gPos, mTerrainBlocks[i]);      GridInfo info;      if (getGridInfo(gPos, info, mTerrainBlocks[i]))      {         // Skip adding this if we already found a GridInfo from the brush terrain         // and the resultant world point isn't equivalent         if (foundBrush)         {            // Convert back to world (since the height can be different)            // Possibly use getHeight() here?            Point3F testWorldPt;            gridToWorld(gPos, testWorldPt, mTerrainBlocks[i]);            if (mFabs( wPos.z - testWorldPt.z ) > 4.0f )               continue;         }         infos.push_back(info);      }   }}void TerrainEditor::setGridInfo(const GridInfo & info, bool checkActive){   PROFILE_SCOPE( TerrainEditor_SetGridInfo );   setGridHeight(info.mGridPoint, info.mHeight);   setGridMaterial(info.mGridPoint, info.mMaterial);}F32 TerrainEditor::getGridHeight(const GridPoint & gPoint){   PROFILE_SCOPE( TerrainEditor_GetGridHeight );   Point2I cPos;   gridToCenter( gPoint.gridPos, cPos );   const TerrainFile *file = gPoint.terrainBlock->getFile();   return fixedToFloat( file->getHeight( cPos.x, cPos.y ) );}void TerrainEditor::gridUpdateComplete( bool materialChanged ){   PROFILE_SCOPE( TerrainEditor_GridUpdateComplete );   // TODO: This updates all terrains and not just the ones   // that were changed.  We should keep track of the mGridUpdate   // in world space and transform it into terrain space.   if(mGridUpdateMin.x <= mGridUpdateMax.x)   {      for (U32 i = 0; i < mTerrainBlocks.size(); i++)      {         TerrainBlock *clientTerrain = getClientTerrain( mTerrainBlocks[i] );         if ( materialChanged )            clientTerrain->updateGridMaterials(mGridUpdateMin, mGridUpdateMax);         mTerrainBlocks[i]->updateGrid(mGridUpdateMin, mGridUpdateMax);         clientTerrain->updateGrid(mGridUpdateMin, mGridUpdateMax);      }   }   mGridUpdateMin.set( S32_MAX, S32_MAX );   mGridUpdateMax.set( 0, 0 );   mNeedsGridUpdate = false;}void TerrainEditor::materialUpdateComplete(){   PROFILE_SCOPE( TerrainEditor_MaterialUpdateComplete );   if(mActiveTerrain && (mGridUpdateMin.x <= mGridUpdateMax.x))   {      TerrainBlock * clientTerrain = getClientTerrain(mActiveTerrain);      clientTerrain->updateGridMaterials(mGridUpdateMin, mGridUpdateMax);   }   mGridUpdateMin.set( S32_MAX, S32_MAX );   mGridUpdateMax.set( 0, 0 );   mNeedsMaterialUpdate = false;}void TerrainEditor::setGridHeight(const GridPoint & gPoint, const F32 height){   PROFILE_SCOPE( TerrainEditor_SetGridHeight );   Point2I cPos;   gridToCenter(gPoint.gridPos, cPos);   mGridUpdateMin.setMin( cPos );   mGridUpdateMax.setMax( cPos );   gPoint.terrainBlock->setHeight(cPos, height);}U8 TerrainEditor::getGridMaterial( const GridPoint &gPoint ) const{   PROFILE_SCOPE( TerrainEditor_GetGridMaterial );   Point2I cPos;   gridToCenter( gPoint.gridPos, cPos );   const TerrainFile *file = gPoint.terrainBlock->getFile();   return file->getLayerIndex( cPos.x, cPos.y );}void TerrainEditor::setGridMaterial( const GridPoint &gPoint, U8 index ){   PROFILE_SCOPE( TerrainEditor_SetGridMaterial );   Point2I cPos;   gridToCenter( gPoint.gridPos, cPos );   TerrainFile *file = gPoint.terrainBlock->getFile();   // If we changed the empty state then we need   // to do a grid update as well.   U8 currIndex = file->getLayerIndex( cPos.x, cPos.y );   if (  ( currIndex == (U8)-1 && index != (U8)-1 ) ||         ( currIndex != (U8)-1 && index == (U8)-1 ) )   {      mGridUpdateMin.setMin( cPos );      mGridUpdateMax.setMax( cPos );      mNeedsGridUpdate = true;   }   file->setLayerIndex( cPos.x, cPos.y, index );}//------------------------------------------------------------------------------TerrainBlock* TerrainEditor::collide(const Gui3DMouseEvent & evt, Point3F & pos){   PROFILE_SCOPE( TerrainEditor_Collide );   if (mTerrainBlocks.size() == 0)      return NULL;   if ( mMouseDown && !String::compare(getCurrentAction(),"paintMaterial") )   {      if ( !mActiveTerrain )         return NULL;      Point3F tpos, tvec;      tpos = evt.pos;      tvec = evt.vec;      mMousePlane.intersect( evt.pos, evt.vec, &pos );      return mActiveTerrain;   }   const U32 mask = TerrainObjectType;   Point3F start( evt.pos );   Point3F end( evt.pos + ( evt.vec * 10000.0f ) );   RayInfo rinfo;   bool hit = gServerContainer.castRay( start, end, mask, &rinfo );   if ( !hit )      return NULL;   pos = rinfo.point;   return (TerrainBlock*)(rinfo.object);   //   //// call the terrain block's ray collision routine directly   //Point3F startPnt = event.pos;   //Point3F endPnt = event.pos + event.vec * 1000.0f;   //S32 blockIndex = -1;   //F32 nearT = 1.0f;   //for (U32 i = 0; i < mTerrainBlocks.size(); i++)   //{   //   Point3F tStartPnt, tEndPnt;   //   mTerrainBlocks[i]->getWorldTransform().mulP(startPnt, &tStartPnt);   //   mTerrainBlocks[i]->getWorldTransform().mulP(endPnt, &tEndPnt);   //   RayInfo ri;   //   if (mTerrainBlocks[i]->castRayI(tStartPnt, tEndPnt, &ri, true))   //   {   //      if (ri.t < nearT)   //      {   //         blockIndex = i;   //         nearT = ri.t;   //      }   //   }   //}   //if (blockIndex > -1)   //{   //   pos.interpolate(startPnt, endPnt, nearT);   //   return mTerrainBlocks[blockIndex];   //}   //return NULL;}//------------------------------------------------------------------------------void TerrainEditor::updateGuiInfo(){   PROFILE_SCOPE( TerrainEditor_UpdateGuiInfo );   char buf[128];   // mouse num grids   // mouse avg height   // selection num grids   // selection avg height   dSprintf(buf, sizeof(buf), "%d %g %g %g %d %g",      mMouseBrush->size(), mMouseBrush->getMinHeight(),      mMouseBrush->getAvgHeight(), mMouseBrush->getMaxHeight(),      mDefaultSel.size(), mDefaultSel.getAvgHeight());   Con::executef(this, "onGuiUpdate", buf);   // If the brush setup has changed send out   // a notification of that!   if ( mBrushChanged && isMethod( "onBrushChanged" ) )   {      mBrushChanged = false;      Con::executef( this, "onBrushChanged" );   }}//------------------------------------------------------------------------------void TerrainEditor::onPreRender(){   PROFILE_SCOPE( TerrainEditor_OnPreRender );   if ( mNeedsGridUpdate )      gridUpdateComplete( mNeedsMaterialUpdate );   else if ( mNeedsMaterialUpdate )      materialUpdateComplete();   Parent::onPreRender();}void TerrainEditor::renderScene(const RectI &){   PROFILE_SCOPE( TerrainEditor_RenderScene );   if(mTerrainBlocks.size() == 0)      return;   if(!mSelectionHidden)      renderSelection(mDefaultSel, LinearColorF::RED, LinearColorF::GREEN, LinearColorF::BLUE, LinearColorF::BLUE, true, false);   if(mRenderBrush && mMouseBrush->size())      renderBrush(*mMouseBrush, LinearColorF::GREEN, LinearColorF::RED, LinearColorF::BLUE, LinearColorF::BLUE, false, true);   if(mRenderBorder)      renderBorder();}//------------------------------------------------------------------------------void TerrainEditor::renderGui( Point2I offset, const RectI &updateRect ){   PROFILE_SCOPE( TerrainEditor_RenderGui );   if ( !mActiveTerrain )      return;   // Just in case...   if ( mMouseBrush->getGridPoint().terrainBlock != mActiveTerrain )      mMouseBrush->setTerrain( mActiveTerrain );   mMouseBrush->render();}void TerrainEditor::renderPoints( const Vector<GFXVertexPCT> &pointList ){   PROFILE_SCOPE( TerrainEditor_RenderPoints );   const U32 pointCount = pointList.size();   const U32 vertCount = pointCount * 6;   GFXStateBlockDesc desc;   desc.setBlend( true );   desc.setZReadWrite( false, false );   GFX->setupGenericShaders();   GFX->setStateBlockByDesc( desc );   U32 vertsLeft = vertCount;   U32 offset = 0;   while ( vertsLeft > 0 )   {      U32 vertsThisDrawCall = getMin( (U32)vertsLeft, (U32)GFX_MAX_DYNAMIC_VERTS );      vertsLeft -= vertsThisDrawCall;      GFXVertexBufferHandle<GFXVertexPCT> vbuff( GFX, vertsThisDrawCall, GFXBufferTypeVolatile );      GFXVertexPCT *vert = vbuff.lock();      const U32 loops = vertsThisDrawCall / 6;      for ( S32 i = 0; i < loops; i++ )      {         const GFXVertexPCT &pointInfo = pointList[i + offset];         vert[0].color = vert[1].color = vert[2].color = vert[3].color = vert[4].color = vert[5].color = pointInfo.color;         const F32 halfSize = pointInfo.texCoord.x * 0.5f;         const Point3F &pos = pointInfo.point;         Point3F p0( pos.x - halfSize, pos.y - halfSize, 0.0f );         Point3F p1( pos.x + halfSize, pos.y - halfSize, 0.0f );         Point3F p2( pos.x + halfSize, pos.y + halfSize, 0.0f );         Point3F p3( pos.x - halfSize, pos.y + halfSize, 0.0f );         vert[0].point = p0;         vert[1].point = p1;         vert[2].point = p2;         vert[3].point = p0;         vert[4].point = p2;         vert[5].point = p3;         vert += 6;      }      vbuff.unlock();      GFX->setVertexBuffer( vbuff );      GFX->drawPrimitive( GFXTriangleList, 0, vertsThisDrawCall / 3 );      offset += loops;   }}//------------------------------------------------------------------------------void TerrainEditor::renderSelection( const Selection & sel, const LinearColorF & inColorFull, const LinearColorF & inColorNone, const LinearColorF & outColorFull, const LinearColorF & outColorNone, bool renderFill, bool renderFrame ){   PROFILE_SCOPE( TerrainEditor_RenderSelection );   // Draw nothing if nothing selected.   if(sel.size() == 0)      return;   Vector<GFXVertexPCT> vertexBuffer;   LinearColorF color;   ColorI iColor;   vertexBuffer.setSize(sel.size() * 5);   F32 squareSize = ( mActiveTerrain ) ? mActiveTerrain->getSquareSize() : 1;   // 'RenderVertexSelection' looks really bad so just always use the good one.   if( false && mRenderVertexSelection)   {      for(U32 i = 0; i < sel.size(); i++)      {         Point3F wPos;         bool center = gridToWorld(sel[i].mGridPoint, wPos);         F32 weight = sel[i].mWeight;         if(center)         {            if ( weight < 0.f || weight > 1.f )               color = inColorFull;            else               color.interpolate( inColorNone, inColorFull, weight );         }         else         {            if ( weight < 0.f || weight > 1.f)               color = outColorFull;            else               color.interpolate( outColorFull, outColorNone, weight );         }         //         iColor = color.toColorI();         GFXVertexPCT *verts = &(vertexBuffer[i * 5]);         verts[0].point = wPos + Point3F(-squareSize, squareSize, 0);         verts[0].color = iColor;         verts[1].point = wPos + Point3F( squareSize, squareSize, 0);         verts[1].color = iColor;         verts[2].point = wPos + Point3F( -squareSize, -squareSize, 0);         verts[2].color = iColor;         verts[3].point = wPos + Point3F( squareSize,  -squareSize, 0);         verts[3].color = iColor;         verts[4].point = verts[0].point;         verts[4].color = iColor;      }   }   else   {      // walk the points in the selection      for(U32 i = 0; i < sel.size(); i++)      {         GridPoint selectedGridPoint = sel[i].mGridPoint;         Point2I gPos = selectedGridPoint.gridPos;         GFXVertexPCT *verts = &(vertexBuffer[i * 5]);         bool center = gridToWorld(selectedGridPoint, verts[0].point);         gridToWorld(Point2I(gPos.x + 1, gPos.y), verts[1].point, selectedGridPoint.terrainBlock);         gridToWorld(Point2I(gPos.x + 1, gPos.y + 1), verts[2].point, selectedGridPoint.terrainBlock);         gridToWorld(Point2I(gPos.x, gPos.y + 1), verts[3].point, selectedGridPoint.terrainBlock);         verts[4].point = verts[0].point;         F32 weight = sel[i].mWeight;         if( !mRenderSolidBrush )         {            if ( center )            {               if ( weight < 0.f || weight > 1.f )                  color = inColorFull;               else                  color.interpolate(inColorNone, inColorFull, weight );            }            else            {               if( weight < 0.f || weight > 1.f )                  color = outColorFull;               else                  color.interpolate(outColorFull, outColorNone, weight );            }            iColor = color.toColorI();         }         else         {            if ( center )            {               iColor = LinearColorF(inColorNone).toColorI();            }            else            {               iColor = LinearColorF(outColorFull).toColorI();            }         }         verts[0].color = iColor;         verts[1].color = iColor;         verts[2].color = iColor;         verts[3].color = iColor;         verts[4].color = iColor;      }   }   // Render this bad boy, by stuffing everything into a volatile buffer   // and rendering...   GFXVertexBufferHandle<GFXVertexPCT> selectionVB(GFX, vertexBuffer.size(), GFXBufferTypeStatic);   selectionVB.lock(0, vertexBuffer.size());   // Copy stuff   dMemcpy((void*)&selectionVB[0], (void*)&vertexBuffer[0], sizeof(GFXVertexPCT) * vertexBuffer.size());   selectionVB.unlock();   GFX->setupGenericShaders();   GFX->setStateBlock( mStateBlock );   GFX->setVertexBuffer(selectionVB);   if(renderFill)      for(U32 i=0; i < sel.size(); i++)         GFX->drawPrimitive( GFXTriangleStrip, i*5, 4);   if(renderFrame)      for(U32 i=0; i < sel.size(); i++)         GFX->drawPrimitive( GFXLineStrip , i*5, 4);}void TerrainEditor::renderBrush( const Brush & brush, const LinearColorF & inColorFull, const LinearColorF & inColorNone, const LinearColorF & outColorFull, const LinearColorF & outColorNone, bool renderFill, bool renderFrame ){}void TerrainEditor::renderBorder(){   // TODO: Disabled rendering the terrain borders... it was   // very annoying getting a fullscreen green tint on things.   //   // We should consider killing this all together or coming   // up with a new technique.   /*   Point2I pos(0,0);   Point2I dir[4] = {      Point2I(1,0),      Point2I(0,1),      Point2I(-1,0),      Point2I(0,-1)   };   GFX->setStateBlock( mStateBlock );   //   if(mBorderLineMode)   {      PrimBuild::color(mBorderFrameColor);      PrimBuild::begin( GFXLineStrip, TerrainBlock::BlockSize * 4 + 1 );      for(U32 i = 0; i < 4; i++)      {         for(U32 j = 0; j < TerrainBlock::BlockSize; j++)         {            Point3F wPos;            gridToWorld(pos, wPos, mActiveTerrain);            PrimBuild::vertex3fv( wPos );            pos += dir[i];         }      }      Point3F wPos;      gridToWorld(Point2I(0,0), wPos, mActiveTerrain);      PrimBuild::vertex3fv( wPos );      PrimBuild::end();   }   else   {      GridSquare * gs = mActiveTerrain->findSquare(TerrainBlock::BlockShift, Point2I(0,0));      F32 height = F32(gs->maxHeight) * 0.03125f + mBorderHeight;      const MatrixF & mat = mActiveTerrain->getTransform();      Point3F pos;      mat.getColumn(3, &pos);      Point2F min(pos.x, pos.y);      Point2F max(pos.x + TerrainBlock::BlockSize * mActiveTerrain->getSquareSize(),                  pos.y + TerrainBlock::BlockSize * mActiveTerrain->getSquareSize());      ColorI & a = mBorderFillColor;      ColorI & b = mBorderFrameColor;      for(U32 i = 0; i < 2; i++)      {         //         if(i){ PrimBuild::color(a); PrimBuild::begin( GFXTriangleFan, 4 ); } else { PrimBuild::color(b); PrimBuild::begin( GFXLineStrip, 5 ); }         PrimBuild::vertex3f(min.x, min.y, 0);         PrimBuild::vertex3f(max.x, min.y, 0);         PrimBuild::vertex3f(max.x, min.y, height);         PrimBuild::vertex3f(min.x, min.y, height);         if(!i) PrimBuild::vertex3f( min.x, min.y, 0.f );         PrimBuild::end();         //         if(i){ PrimBuild::color(a); PrimBuild::begin( GFXTriangleFan, 4 ); } else { PrimBuild::color(b); PrimBuild::begin( GFXLineStrip, 5 ); }         PrimBuild::vertex3f(min.x, max.y, 0);         PrimBuild::vertex3f(max.x, max.y, 0);         PrimBuild::vertex3f(max.x, max.y, height);         PrimBuild::vertex3f(min.x, max.y, height);         if(!i) PrimBuild::vertex3f( min.x, min.y, 0.f );         PrimBuild::end();         //         if(i){ PrimBuild::color(a); PrimBuild::begin( GFXTriangleFan, 4 ); } else { PrimBuild::color(b); PrimBuild::begin( GFXLineStrip, 5 ); }         PrimBuild::vertex3f(min.x, min.y, 0);         PrimBuild::vertex3f(min.x, max.y, 0);         PrimBuild::vertex3f(min.x, max.y, height);         PrimBuild::vertex3f(min.x, min.y, height);         if(!i) PrimBuild::vertex3f( min.x, min.y, 0.f );         PrimBuild::end();         //         if(i){ PrimBuild::color(a); PrimBuild::begin( GFXTriangleFan, 4 ); } else { PrimBuild::color(b); PrimBuild::begin( GFXLineStrip, 5 ); }         PrimBuild::vertex3f(max.x, min.y, 0);         PrimBuild::vertex3f(max.x, max.y, 0);         PrimBuild::vertex3f(max.x, max.y, height);         PrimBuild::vertex3f(max.x, min.y, height);         if(!i) PrimBuild::vertex3f( min.x, min.y, 0.f );         PrimBuild::end();      }   }   */}void TerrainEditor::submitUndo( Selection *sel ){   // Grab the mission editor undo manager.   UndoManager *undoMan = NULL;   if ( !Sim::findObject( "EUndoManager", undoMan ) )   {      Con::errorf( "TerrainEditor::submitUndo() - EUndoManager not found!" );      return;   }   // Create and submit the action.   TerrainEditorUndoAction *action = new TerrainEditorUndoAction( "Terrain Editor Action" );   action->mSel = sel;   action->mTerrainEditor = this;   undoMan->addAction( action );   // Mark the editor as dirty!   setDirty();}void TerrainEditor::TerrainEditorUndoAction::undo(){   // NOTE: This function also handles TerrainEditorUndoAction::redo().   bool materialChanged = false;   for (U32 i = 0; i < mSel->size(); i++)   {      // Grab the current grid info for this point.      GridInfo info;      mTerrainEditor->getGridInfo( (*mSel)[i].mGridPoint, info );      info.mMaterialChanged = (*mSel)[i].mMaterialChanged;      materialChanged |= info.mMaterialChanged;      // Restore the previous grid info.      mTerrainEditor->setGridInfo( (*mSel)[i] );      // Save the old grid info so we can      // restore it later.      (*mSel)[i] = info;   }   // Mark the editor as dirty!   mTerrainEditor->setDirty();   mTerrainEditor->gridUpdateComplete( materialChanged );   mTerrainEditor->mMouseBrush->update();}void TerrainEditor::submitMaterialUndo( String actionName ){   // Grab the mission editor undo manager.   UndoManager *undoMan = NULL;   if ( !Sim::findObject( "EUndoManager", undoMan ) )   {      Con::errorf( "TerrainEditor::submitMaterialUndo() - EUndoManager not found!" );      return;   }   TerrainBlock *terr = getClientTerrain();   // Create and submit the action.   TerrainMaterialUndoAction *action = new TerrainMaterialUndoAction( actionName );   action->mTerrain = terr;   action->mMaterials = terr->getMaterials();   action->mLayerMap = terr->getLayerMap();   action->mEditor = this;   undoMan->addAction( action );   // Mark the editor as dirty!   setDirty();}void TerrainEditor::onMaterialUndo( TerrainBlock *terr ){   setDirty();   scheduleMaterialUpdate();   setGridUpdateMinMax();   terr->mDetailsDirty = true;   terr->mLayerTexDirty = true;   Con::executef( this, "onMaterialUndo" );}void TerrainEditor::TerrainMaterialUndoAction::undo(){   Vector<TerrainMaterial*> tempMaterials = mTerrain->getMaterials();   Vector<U8> tempLayers = mTerrain->getLayerMap();   mTerrain->setMaterials(mMaterials);   mTerrain->setLayerMap(mLayerMap);   mMaterials = tempMaterials;   mLayerMap = tempLayers;   mEditor->onMaterialUndo( mTerrain );}void TerrainEditor::TerrainMaterialUndoAction::redo(){   undo();}class TerrainProcessActionEvent : public SimEvent{   U32 mSequence;public:   TerrainProcessActionEvent(U32 seq)   {      mSequence = seq;   }   void process(SimObject *object) override   {      ((TerrainEditor *) object)->processActionTick(mSequence);   }};void TerrainEditor::processActionTick(U32 sequence){   if(mMouseDownSeq == sequence)   {      Sim::postEvent(this, new TerrainProcessActionEvent(mMouseDownSeq), Sim::getCurrentTime() + 30);      mCurrentAction->process(mMouseBrush, mLastEvent, false, TerrainAction::Update);   }}bool TerrainEditor::onInputEvent(const InputEventInfo & event){   /*   if (  mRightMousePassThru &&         event.deviceType == KeyboardDeviceType &&         event.objType == SI_KEY &&         event.objInst == KEY_TAB &&         event.action == SI_MAKE )   {      if ( isMethod( "onToggleToolWindows" ) )         Con::executef( this, "onToggleToolWindows" );   }   */   return Parent::onInputEvent( event );}void TerrainEditor::on3DMouseDown(const Gui3DMouseEvent & event){   getRoot()->showCursor( false );   if(mTerrainBlocks.size() == 0)      return;   if (!String::compare(getCurrentAction(),"paintMaterial"))   {      Point3F pos;      TerrainBlock* hitTerrain = collide(event, pos);      if(!hitTerrain)         return;      // Set the active terrain      bool changed = mActiveTerrain != hitTerrain;      mActiveTerrain = hitTerrain;      if (changed)      {         Con::executef(this, "onActiveTerrainChange", Con::getIntArg(hitTerrain->getId()));         mMouseBrush->setTerrain(mActiveTerrain);         //if(mRenderBrush)            //mCursorVisible = false;         mMousePos = pos;         mMouseBrush->setPosition(mMousePos);         return;      }   }   else if ((event.modifier & SI_ALT) && !String::compare(getCurrentAction(),"setHeight"))   {      // Set value to terrain height at mouse position      GridInfo info;      getGridInfo(mMouseBrush->getGridPoint(), info);      mSetHeightVal = info.mHeight;      mBrushChanged = true;      return;   }   mMousePlane.set( mMousePos, Point3F(0,0,1) );   mMouseDown = true;   mSelectionLocked = false;   mouseLock();   mMouseDownSeq++;   mUndoSel = new Selection;   mCurrentAction->process(mMouseBrush, event, true, TerrainAction::Begin);   // process on ticks - every 30th of a second.   Sim::postEvent(this, new TerrainProcessActionEvent(mMouseDownSeq), Sim::getCurrentTime() + 30);}void TerrainEditor::on3DMouseMove(const Gui3DMouseEvent & event){   PROFILE_SCOPE( TerrainEditor_On3DMouseMove );   if(mTerrainBlocks.size() == 0)      return;   Point3F pos;   TerrainBlock* hitTerrain = collide(event, pos);   if(!hitTerrain)   {      mMouseBrush->reset();      return;   }   else   {      // We do not change the active terrain as the mouse moves when      // in painting mode.  This is because it causes the material      // window to change as you cursor over to it.      if ( String::compare(getCurrentAction(),"paintMaterial") != 0 )      {         // Set the active terrain         bool changed = mActiveTerrain != hitTerrain;         mActiveTerrain = hitTerrain;         if (changed)            Con::executef(this, "onActiveTerrainChange", Con::getIntArg(hitTerrain->getId()));      }      mMousePos = pos;      mMouseBrush->setTerrain(mActiveTerrain);      mMouseBrush->setPosition(mMousePos);   }}void TerrainEditor::on3DMouseDragged(const Gui3DMouseEvent & event){   PROFILE_SCOPE( TerrainEditor_On3DMouseDragged );   if ( mTerrainBlocks.empty() )      return;   if ( !isMouseLocked() )      return;   Point3F pos;   bool selChanged = false;   if ( !mSelectionLocked )   {      TerrainBlock* hitTerrain = collide(event, pos);      if (!hitTerrain)      {         mMouseBrush->reset();         return;      }      // check if the mouse has actually moved in grid space      Point2I gMouse;      Point2I gLastMouse;      worldToGrid( pos, gMouse );      worldToGrid( mMousePos, gLastMouse );      mMousePos = pos;      mMouseBrush->setPosition( mMousePos );      selChanged = gMouse != gLastMouse;   }   if (selChanged)      mCurrentAction->process( mMouseBrush, event, true, TerrainAction::Update );}void TerrainEditor::on3DMouseUp(const Gui3DMouseEvent & event){   mMouseDown = false;   getRoot()->showCursor( true );   if ( mTerrainBlocks.size() == 0 )      return;   if ( isMouseLocked() )   {      mouseUnlock();      mMouseDownSeq++;      mCurrentAction->process( mMouseBrush, event, false, TerrainAction::End );      if ( mUndoSel->size() )         submitUndo( mUndoSel );      else         delete mUndoSel;      mUndoSel = 0;      mInAction = false;   }}bool TerrainEditor::onMouseWheelDown( const GuiEvent & event ){   if ( event.modifier & SI_PRIMARY_CTRL && event.modifier & SI_SHIFT )   {      setBrushPressure( mBrushPressure - 0.1f );      return true;   }   else if ( event.modifier & SI_SHIFT )   {      setBrushSoftness( mBrushSoftness + 0.05f );      return true;   }   else if ( event.modifier & SI_PRIMARY_CTRL )   {      Point2I newBrush = getBrushSize() - Point2I(1,1);      setBrushSize( newBrush.x, newBrush.y );      return true;   }   return Parent::onMouseWheelDown( event );}bool TerrainEditor::onMouseWheelUp( const GuiEvent & event ){   if ( event.modifier & SI_PRIMARY_CTRL && event.modifier & SI_SHIFT )   {      setBrushPressure( mBrushPressure + 0.1f );      return true;   }   else if ( event.modifier & SI_SHIFT )   {      setBrushSoftness( mBrushSoftness - 0.05f );      return true;   }   else if( event.modifier & SI_PRIMARY_CTRL )   {      Point2I newBrush = getBrushSize() + Point2I(1,1);      setBrushSize( newBrush.x, newBrush.y );      return true;   }   return Parent::onMouseWheelUp( event );}//------------------------------------------------------------------------------// any console function which depends on a terrainBlock attached to the editor// should call thisbool checkTerrainBlock(TerrainEditor * object, const char * funcName){   if(!object->terrainBlockValid())   {      Con::errorf(ConsoleLogEntry::Script, "TerrainEditor::%s: not attached to a terrain block!", funcName);      return(false);   }   return(true);}void TerrainEditor::attachTerrain(TerrainBlock *terrBlock){   mActiveTerrain = terrBlock;   mTerrainBlocks.push_back_unique(terrBlock);}void TerrainEditor::detachTerrain(TerrainBlock *terrBlock){   if (mActiveTerrain == terrBlock)      mActiveTerrain = NULL; //do we want to set this to an existing terrain?   if (mMouseBrush->getGridPoint().terrainBlock == terrBlock)      mMouseBrush->setTerrain(NULL);   // reset the brush as its gridinfos may still have references to the old terrain   mMouseBrush->reset();   mTerrainBlocks.remove(terrBlock);}TerrainBlock* TerrainEditor::getTerrainBlock(S32 index){   if(index < 0 || index >= mTerrainBlocks.size())      return NULL;   return mTerrainBlocks[index];}void TerrainEditor::getTerrainBlocksMaterialList(Vector<StringTableEntry>& list){   for(S32 i=0; i<mTerrainBlocks.size(); ++i)   {      TerrainBlock* tb = mTerrainBlocks[i];      if(!tb)         continue;      for(S32 m=0; m<tb->getMaterialCount(); ++m)      {         TerrainMaterial* mat = tb->getMaterial(m);         if (mat)            list.push_back_unique(mat->getInternalName());      }   }}void TerrainEditor::setBrushType( const char *type ){   if ( mMouseBrush && String::compare( mMouseBrush->getType(), type ) == 0 )      return;   if(!dStricmp(type, "box"))   {      delete mMouseBrush;      mMouseBrush = new BoxBrush(this);      mBrushChanged = true;   }   else if(!dStricmp(type, "ellipse"))   {      delete mMouseBrush;      mMouseBrush = new EllipseBrush(this);      mBrushChanged = true;   }   else if(!dStricmp(type, "selection"))   {      delete mMouseBrush;      mMouseBrush = new SelectionBrush(this);      mBrushChanged = true;   }   else {}}const char* TerrainEditor::getBrushType() const{   if ( mMouseBrush )      return mMouseBrush->getType();   return "";}void TerrainEditor::setBrushSize( S32 w, S32 h ){   w = mClamp( w, 1, mMaxBrushSize.x );   h = mClamp( h, 1, mMaxBrushSize.y );   if ( w == mBrushSize.x && h == mBrushSize.y )      return;	mBrushSize.set( w, h );   mBrushChanged = true;   if ( mMouseBrush )   {   	mMouseBrush->setSize( mBrushSize );      if ( mMouseBrush->getGridPoint().terrainBlock )         mMouseBrush->rebuild();   }}void TerrainEditor::setBrushPressure( F32 pressure ){   pressure = mClampF( pressure, 0.01f, 1.0f );   if ( mBrushPressure == pressure )      return;   mBrushPressure = pressure;   mBrushChanged = true;   if ( mMouseBrush && mMouseBrush->getGridPoint().terrainBlock )      mMouseBrush->rebuild();}void TerrainEditor::setBrushSoftness( F32 softness ){   softness = mClampF( softness, 0.01f, 1.0f );   if ( mBrushSoftness == softness )      return;   mBrushSoftness = softness;   mBrushChanged = true;   if ( mMouseBrush && mMouseBrush->getGridPoint().terrainBlock )      mMouseBrush->rebuild();}const char* TerrainEditor::getBrushPos(){   AssertFatal(mMouseBrush!=NULL, "TerrainEditor::getBrushPos: no mouse brush!");   Point2I pos = mMouseBrush->getPosition();   static const U32 bufSize = 32;   char * ret = Con::getReturnBuffer(bufSize);   dSprintf(ret, bufSize, "%d %d", pos.x, pos.y);   return(ret);}void TerrainEditor::setBrushPos(Point2I pos){   AssertFatal(mMouseBrush!=NULL, "TerrainEditor::setBrushPos: no mouse brush!");   mMouseBrush->setPosition(pos);}void TerrainEditor::setAction(const char* action){   for(U32 i = 0; i < mActions.size(); i++)   {      if(!dStricmp(mActions[i]->getName(), action))      {         mCurrentAction = mActions[i];         //         mRenderBrush = mCurrentAction->useMouseBrush();         return;      }   }}const char* TerrainEditor::getActionName(U32 index){   if(index >= mActions.size())      return("");   return(mActions[index]->getName());}const char* TerrainEditor::getCurrentAction() const{   return(mCurrentAction->getName());}S32 TerrainEditor::getNumActions(){   return(mActions.size());}void TerrainEditor::resetSelWeights(bool clear){   //   if(!clear)   {      for(U32 i = 0; i < mDefaultSel.size(); i++)      {         mDefaultSel[i].mPrimarySelect = false;         mDefaultSel[i].mWeight = 1.f;      }      return;   }   Selection sel;   U32 i;   for(i = 0; i < mDefaultSel.size(); i++)   {      if(mDefaultSel[i].mPrimarySelect)      {         mDefaultSel[i].mWeight = 1.f;         sel.add(mDefaultSel[i]);      }   }   mDefaultSel.reset();   for(i = 0; i < sel.size(); i++)      mDefaultSel.add(sel[i]);}void TerrainEditor::clearSelection(){	mDefaultSel.reset();}void TerrainEditor::processAction(const char* sAction){   if(!checkTerrainBlock(this, "processAction"))      return;   TerrainAction * action = mCurrentAction;   if (String::compare(sAction, "") != 0)   {      action = lookupAction(sAction);      if(!action)      {         Con::errorf(ConsoleLogEntry::General, "TerrainEditor::cProcessAction: invalid action name '%s'.", sAction);         return;      }   }   if(!getCurrentSel()->size() && !mProcessUsesBrush)      return;   mUndoSel = new Selection;   Gui3DMouseEvent event;   if(mProcessUsesBrush)      action->process(mMouseBrush, event, true, TerrainAction::Process);   else      action->process(getCurrentSel(), event, true, TerrainAction::Process);   // check if should delete the undo   if(mUndoSel->size())      submitUndo( mUndoSel );   else      delete mUndoSel;   mUndoSel = 0;}S32 TerrainEditor::getNumTextures(){   if(!checkTerrainBlock(this, "getNumTextures"))      return(0);   // walk all the possible material lists and count them..   U32 count = 0;   for (U32 t = 0; t < mTerrainBlocks.size(); t++)      count += mTerrainBlocks[t]->getMaterialCount();   return count;}void TerrainEditor::markEmptySquares(){   if(!checkTerrainBlock(this, "markEmptySquares"))      return;}void TerrainEditor::mirrorTerrain(S32 mirrorIndex){   if(!checkTerrainBlock(this, "mirrorTerrain"))      return;   // TODO!   /*   TerrainBlock * terrain = mActiveTerrain;   setDirty();   //   enum {      top = BIT(0),      bottom = BIT(1),      left = BIT(2),      right = BIT(3)   };   U32 sides[8] =   {      bottom,      bottom | left,      left,      left | top,      top,      top | right,      right,      bottom | right   };   U32 n = TerrainBlock::BlockSize;   U32 side = sides[mirrorIndex % 8];   bool diag = mirrorIndex & 0x01;   Point2I src((side & right) ? (n - 1) : 0, (side & bottom) ? (n - 1) : 0);   Point2I dest((side & left) ? (n - 1) : 0, (side & top) ? (n - 1) : 0);   Point2I origSrc(src);   Point2I origDest(dest);   // determine the run length   U32 minStride = ((side & top) || (side & bottom)) ? n : n / 2;   U32 majStride = ((side & left) || (side & right)) ? n : n / 2;   Point2I srcStep((side & right) ? -1 : 1, (side & bottom) ? -1 : 1);   Point2I destStep((side & left) ? -1 : 1, (side & top) ? -1 : 1);   //   U16 * heights = terrain->getHeightAddress(0,0);   U8 * baseMaterials = terrain->getBaseMaterialAddress(0,0);   TerrainBlock::Material * materials = terrain->getMaterial(0,0);   // create an undo selection   Selection * undo = new Selection;   // walk through all the positions   for(U32 i = 0; i < majStride; i++)   {      for(U32 j = 0; j < minStride; j++)      {         // skip the same position         if(src != dest)         {            U32 si = src.x + (src.y << TerrainBlock::BlockShift);            U32 di = dest.x + (dest.y << TerrainBlock::BlockShift);            // add to undo selection            GridInfo info;            getGridInfo(dest, info, terrain);            undo->add(info);            //... copy info... (height, basematerial, material)            heights[di] = heights[si];            baseMaterials[di] = baseMaterials[si];            materials[di] = materials[si];         }         // get to the new position         src.x += srcStep.x;         diag ? (dest.y += destStep.y) : (dest.x += destStep.x);      }      // get the next position for a run      src.y += srcStep.y;      diag ? (dest.x += destStep.x) : (dest.y += destStep.y);      // reset the minor run      src.x = origSrc.x;      diag ? (dest.y = origDest.y) : (dest.x = origDest.x);      // shorten the run length for diag runs      if(diag)         minStride--;   }   // rebuild stuff..   terrain->buildGridMap();   terrain->rebuildEmptyFlags();   terrain->packEmptySquares();   // add undo selection   submitUndo( undo );   */}bool TerrainEditor::isPointInTerrain( const GridPoint & gPoint){   PROFILE_SCOPE( TerrainEditor_IsPointInTerrain );   Point2I cPos;   gridToCenter( gPoint.gridPos, cPos );   const TerrainFile *file = gPoint.terrainBlock->getFile();   return file->isPointInTerrain( cPos.x, cPos.y );}void TerrainEditor::reorderMaterial( S32 index, S32 orderPos ){   TerrainBlock *terr = getClientTerrain();   Vector<U8> layerMap = terr->getLayerMap();   Vector<TerrainMaterial*> materials = terr->getMaterials();   TerrainMaterial *pMat = materials[index];   submitMaterialUndo( String::ToString( "Reordered %s Material", terr->getMaterialName(index) ) );   materials.erase( index );   materials.insert( orderPos, pMat );   Vector<U8>::iterator itr = layerMap.begin();   for ( ; itr != layerMap.end(); itr++ )   {      // Was previous material, set to new index.      if ( *itr == index )         *itr = orderPos;      else      {         // We removed a Material prior to this one, bump it down.         if ( *itr > index )            (*itr)--;         // We added a Material prior to this one, bump it up.         if ( *itr >= orderPos )            (*itr)++;      }   }   terr->setMaterials( materials );   terr->setLayerMap( layerMap );   // We didn't really just "undo" but it happens to do everything we   // need to update the materials and gui.   onMaterialUndo( terr );}//------------------------------------------------------------------------------DefineEngineMethod( TerrainEditor, attachTerrain, void, (const char * terrain), (""), "(TerrainBlock terrain)"){   Scene* scene = Scene::getRootScene();   if (!scene)   {      Con::errorf(ConsoleLogEntry::Script, "TerrainEditor::attach: no scene found");      return;   }   VectorPtr<TerrainBlock*> terrains;   // attach to first found terrainBlock   if (String::compare (terrain,"")==0)   {      for(SimSetIterator itr(scene); *itr; ++itr)      {         TerrainBlock* terrBlock = dynamic_cast<TerrainBlock*>(*itr);         if (terrBlock)            terrains.push_back(terrBlock);      }      //if (terrains.size() == 0)      //   Con::errorf(ConsoleLogEntry::Script, "TerrainEditor::attach: no TerrainBlock objects found!");   }   else  // attach to named object   {      TerrainBlock* terrBlock = dynamic_cast<TerrainBlock*>(Sim::findObject(terrain));      if (terrBlock)         terrains.push_back(terrBlock);      if(terrains.size() == 0)         Con::errorf(ConsoleLogEntry::Script, "TerrainEditor::attach: failed to attach to object '%s'", terrain);   }   if (terrains.size() > 0)   {      for (U32 i = 0; i < terrains.size(); i++)      {         if (!terrains[i]->isServerObject())         {            terrains[i] = NULL;            Con::errorf(ConsoleLogEntry::Script, "TerrainEditor::attach: cannot attach to client TerrainBlock");         }      }   }   for (U32 i = 0; i < terrains.size(); i++)   {      if (terrains[i])	      object->attachTerrain(terrains[i]);   }}DefineEngineMethod( TerrainEditor, getTerrainBlockCount, S32, (), , "()"){   return object->getTerrainBlockCount();}DefineEngineMethod( TerrainEditor, getTerrainBlock, S32, (S32 index), , "(S32 index)"){   TerrainBlock* tb = object->getTerrainBlock(index);   if(!tb)      return 0;   else      return tb->getId();}DefineEngineMethod(TerrainEditor, getTerrainBlocksMaterialList, const char *, (), , "() gets the list of current terrain materials for all terrain blocks."){   Vector<StringTableEntry> list;   object->getTerrainBlocksMaterialList(list);   if(list.size() == 0)      return "";   // Calculate the size of the return buffer   S32 size = 0;   for(U32 i = 0; i < list.size(); ++i)   {      size += dStrlen(list[i]);      ++size;   }   ++size;   // Copy the material names   char *ret = Con::getReturnBuffer(size);   ret[0] = 0;   for(U32 i = 0; i < list.size(); ++i)   {      dStrcat( ret, list[i], size );      dStrcat( ret, "\n", size );   }   return ret;}DefineEngineMethod( TerrainEditor, setBrushType, void, (String type), , "(string type)"              "One of box, ellipse, selection."){	object->setBrushType(type);}DefineEngineMethod( TerrainEditor, getBrushType, const char*, (), , "()"){   return object->getBrushType();}DefineEngineMethod( TerrainEditor, setBrushSize, void, ( S32 w, S32 h), (0), "(int w [, int h])"){	object->setBrushSize( w, h==0?w:h );}DefineEngineMethod( TerrainEditor, getBrushSize, const char*, (), , "()"){   Point2I size = object->getBrushSize();   static const U32 bufSize = 32;   char * ret = Con::getReturnBuffer(bufSize);   dSprintf(ret, bufSize, "%d %d", size.x, size.y);   return ret;}DefineEngineMethod( TerrainEditor, setBrushPressure, void, (F32 pressure), , "(float pressure)"){   object->setBrushPressure( pressure );}DefineEngineMethod( TerrainEditor, getBrushPressure, F32, (), , "()"){   return object->getBrushPressure();}DefineEngineMethod( TerrainEditor, setBrushSoftness, void, (F32 softness), , "(float softness)"){   object->setBrushSoftness( softness );}DefineEngineMethod( TerrainEditor, getBrushSoftness, F32, (), , "()"){   return object->getBrushSoftness();}DefineEngineMethod( TerrainEditor, getBrushPos, const char*, (), , "Returns a Point2I."){	return object->getBrushPos();}DefineEngineMethod( TerrainEditor, setBrushPos, void, (Point2I pos), , "Location"){   object->setBrushPos(pos);}DefineEngineMethod( TerrainEditor, setAction, void, (const char * action_name), , "(string action_name)"){	object->setAction(action_name);}DefineEngineMethod( TerrainEditor, getActionName, const char*, (U32 index), , "(int num)"){	return (object->getActionName(index));}DefineEngineMethod( TerrainEditor, getNumActions, S32, (), , ""){	return(object->getNumActions());}DefineEngineMethod( TerrainEditor, getCurrentAction, const char*, (), , ""){	return object->getCurrentAction();}DefineEngineMethod( TerrainEditor, resetSelWeights, void, (bool clear), , "(bool clear)"){	object->resetSelWeights(clear);}DefineEngineMethod( TerrainEditor, clearSelection, void, (), , ""){   object->clearSelection();}DefineEngineMethod( TerrainEditor, processAction, void, (String action), (""), "(string action=NULL)"){	object->processAction(action);}DefineEngineMethod( TerrainEditor, getActiveTerrain, S32, (), , ""){   S32 ret = 0;   TerrainBlock* terrain = object->getActiveTerrain();   if (terrain)      ret = terrain->getId();	return ret;}DefineEngineMethod( TerrainEditor, getNumTextures, S32, (), , ""){	return object->getNumTextures();}DefineEngineMethod( TerrainEditor, markEmptySquares, void, (), , ""){	object->markEmptySquares();}DefineEngineMethod( TerrainEditor, mirrorTerrain, void, (S32 mirrorIndex), , ""){	object->mirrorTerrain(mirrorIndex);}DefineEngineMethod(TerrainEditor, setTerraformOverlay, void, (bool overlayEnable), , "(bool overlayEnable) - sets the terraformer current heightmap to draw as an overlay over the current terrain."){   // XA: This one needs to be implemented :)}DefineEngineMethod(TerrainEditor, updateMaterial, bool, ( U32 index, String matName ), ,   "( int index, string matName )\n"   "Changes the material name at the index." ){   TerrainBlock *terr = object->getClientTerrain();   if ( !terr )      return false;   if ( index >= terr->getMaterialCount() )      return false;   terr->updateMaterial( index, matName );   object->setDirty();   return true;}DefineEngineMethod(TerrainEditor, addMaterial, S32, ( String matName ), ,   "( string matName )\n"   "Adds a new material." ){   TerrainBlock *terr = object->getClientTerrain();   if ( !terr )      return false;   terr->addMaterial( matName );   object->setDirty();   return true;}DefineEngineMethod( TerrainEditor, removeMaterial, void, ( S32 index ), , "( int index ) - Remove the material at the given index." ){   TerrainBlock *terr = object->getClientTerrain();   if ( !terr )      return;   if ( index < 0 || index >= terr->getMaterialCount() )   {      Con::errorf( "TerrainEditor::removeMaterial - index out of range!" );      return;   }   if ( terr->getMaterialCount() == 1 )   {      Con::errorf( "TerrainEditor::removeMaterial - cannot remove material, there is only one!" );      return;   }   const char *matName = terr->getMaterialName( index );   object->submitMaterialUndo( String::ToString( "Remove TerrainMaterial %s", matName ) );   terr->removeMaterial( index );   object->setDirty();   object->scheduleMaterialUpdate();   object->setGridUpdateMinMax();}DefineEngineMethod(TerrainEditor, getMaterialCount, S32, (), ,   "Returns the current material count." ){   TerrainBlock *terr = object->getClientTerrain();   if ( terr )      return terr->getMaterialCount();   return 0;}DefineEngineMethod(TerrainEditor, getMaterials, const char *, (), , "() gets the list of current terrain materials."){   TerrainBlock *terr = object->getClientTerrain();   if ( !terr )      return "";   char *ret = Con::getReturnBuffer(4096);   ret[0] = 0;   for(U32 i = 0; i < terr->getMaterialCount(); i++)   {      dStrcat( ret, terr->getMaterialName(i), 4096 );      dStrcat( ret, "\n", 4096 );   }   return ret;}DefineEngineMethod( TerrainEditor, getMaterialName, const char*, (S32 index), , "( int index ) - Returns the name of the material at the given index." ){   TerrainBlock *terr = object->getClientTerrain();   if ( !terr )      return "";   if( index < 0 || index >= terr->getMaterialCount() )   {      Con::errorf( "TerrainEditor::getMaterialName - index out of range!" );      return "";   }   const char* name = terr->getMaterialName( index );   return Con::getReturnBuffer( name );}DefineEngineMethod( TerrainEditor, getMaterialIndex, S32, ( String name ), , "( string name ) - Returns the index of the material with the given name or -1." ){   TerrainBlock *terr = object->getClientTerrain();   if ( !terr )      return -1;   const U32 count = terr->getMaterialCount();   for( U32 i = 0; i < count; ++ i )      if( dStricmp( name, terr->getMaterialName( i ) ) == 0 )         return i;   return -1;}DefineEngineMethod( TerrainEditor, reorderMaterial, void, ( S32 index, S32 orderPos ), , "( int index, int order ) "  "- Reorder material at the given index to the new position, changing the order in which it is rendered / blended." ){   object->reorderMaterial( index, orderPos );}DefineEngineMethod(TerrainEditor, getTerrainUnderWorldPoint, S32, (const char * ptOrX, const char * Y, const char * Z), ("", "", ""),                                                                           "(x/y/z) Gets the terrain block that is located under the given world point.\n"                                                                           "@param x/y/z The world coordinates (floating point values) you wish to query at. "                                                                           "These can be formatted as either a string (\"x y z\") or separately as (x, y, z)\n"                                                                           "@return Returns the ID of the requested terrain block (0 if not found).\n\n"){   TerrainEditor *tEditor = (TerrainEditor *) object;   if(tEditor == NULL)      return 0;   Point3F pos;   if(!String::isEmpty(ptOrX) && String::isEmpty(Y) && String::isEmpty(Z))      dSscanf(ptOrX, "%f %f %f", &pos.x, &pos.y, &pos.z);   else if(!String::isEmpty(ptOrX) && !String::isEmpty(Y) && !String::isEmpty(Z))   {      pos.x = dAtof(ptOrX);      pos.y = dAtof(Y);      pos.z = dAtof(Z);   }   else   {      Con::errorf("TerrainEditor.getTerrainUnderWorldPoint(): Invalid argument count! Valid arguments are either \"x y z\" or x,y,z\n");      return 0;   }   TerrainBlock* terrain = tEditor->getTerrainUnderWorldPoint(pos);   if(terrain != NULL)   {      return terrain->getId();   }   return 0;}//------------------------------------------------------------------------------void TerrainEditor::initPersistFields(){   docsURL;   addGroup("Misc");   addField("isDirty", TypeBool, Offset(mIsDirty, TerrainEditor));   addField("isMissionDirty", TypeBool, Offset(mIsMissionDirty, TerrainEditor));   addField("renderBorder", TypeBool, Offset(mRenderBorder, TerrainEditor));                    ///< Not currently used   addField("borderHeight", TypeF32, Offset(mBorderHeight, TerrainEditor));                     ///< Not currently used   addField("borderFillColor", TypeColorI, Offset(mBorderFillColor, TerrainEditor));            ///< Not currently used   addField("borderFrameColor", TypeColorI, Offset(mBorderFrameColor, TerrainEditor));          ///< Not currently used   addField("borderLineMode", TypeBool, Offset(mBorderLineMode, TerrainEditor));                ///< Not currently used   addField("selectionHidden", TypeBool, Offset(mSelectionHidden, TerrainEditor));   addField("renderVertexSelection", TypeBool, Offset(mRenderVertexSelection, TerrainEditor));  ///< Not currently used   addField("renderSolidBrush", TypeBool, Offset(mRenderSolidBrush, TerrainEditor));   addField("processUsesBrush", TypeBool, Offset(mProcessUsesBrush, TerrainEditor));   addField("maxBrushSize", TypePoint2I, Offset(mMaxBrushSize, TerrainEditor));   // action values...   addField("adjustHeightVal", TypeF32, Offset(mAdjustHeightVal, TerrainEditor));               ///< RaiseHeightAction and LowerHeightAction   addField("setHeightVal", TypeF32, Offset(mSetHeightVal, TerrainEditor));                     ///< SetHeightAction   addField("scaleVal", TypeF32, Offset(mScaleVal, TerrainEditor));                             ///< ScaleHeightAction   addField("smoothFactor", TypeF32, Offset(mSmoothFactor, TerrainEditor));                     ///< SmoothHeightAction   addField("noiseFactor", TypeF32, Offset(mNoiseFactor, TerrainEditor));                       ///< PaintNoiseAction   addField("materialGroup", TypeS32, Offset(mMaterialGroup, TerrainEditor));                   ///< Not currently used   addField("softSelectRadius", TypeF32, Offset(mSoftSelectRadius, TerrainEditor));             ///< SoftSelectAction   addField("softSelectFilter", TypeString, Offset(mSoftSelectFilter, TerrainEditor));          ///< SoftSelectAction brush filtering   addField("softSelectDefaultFilter", TypeString, Offset(mSoftSelectDefaultFilter, TerrainEditor));  ///< SoftSelectAction brush filtering   addField("adjustHeightMouseScale", TypeF32, Offset(mAdjustHeightMouseScale, TerrainEditor)); ///< Not currently used   addField("paintIndex", TypeS32, Offset(mPaintIndex, TerrainEditor));                         ///< PaintMaterialAction   endGroup("Misc");   Parent::initPersistFields();}DefineEngineMethod( TerrainEditor, getSlopeLimitMinAngle, F32, (), , ""){   return object->mSlopeMinAngle;}DefineEngineMethod( TerrainEditor, setSlopeLimitMinAngle, F32, (F32 angle), , ""){	if ( angle < 0.0f )		angle = 0.0f;   if ( angle > object->mSlopeMaxAngle )      angle = object->mSlopeMaxAngle;	object->mSlopeMinAngle = angle;	return angle;}DefineEngineMethod( TerrainEditor, getSlopeLimitMaxAngle, F32, (), , ""){   return object->mSlopeMaxAngle;}DefineEngineMethod( TerrainEditor, setSlopeLimitMaxAngle, F32, (F32 angle), , ""){	if ( angle > 90.0f )		angle = 90.0f;   if ( angle < object->mSlopeMinAngle )      angle = object->mSlopeMinAngle;	object->mSlopeMaxAngle = angle;	return angle;}DefineEngineMethod(TerrainEditor, getTileLimitMinHeight, F32, (), , ""){   return object->mTileMinHeight;}DefineEngineMethod(TerrainEditor, setTileLimitMinHeight, F32, (F32 height), , ""){   if (height < 0.0f)      height = 0.0f;   if (height > object->mTileMaxHeight)      height = object->mTileMaxHeight;   object->mTileMinHeight = height;   return height;}DefineEngineMethod(TerrainEditor, getTileLimitMaxHeight, F32, (), , ""){   return object->mTileMaxHeight;}DefineEngineMethod(TerrainEditor, setTileLimitMaxHeight, F32, (F32 height), , ""){   if (height > 2047.0f)      height = 2047.0f;   if (height < object->mTileMinHeight)      height = object->mTileMinHeight;   object->mTileMaxHeight = height;   return height;}//------------------------------------------------------------------------------void TerrainEditor::autoMaterialLayer( F32 mMinHeight, F32 mMaxHeight, F32 mMinSlope, F32 mMaxSlope, F32 mCoverage ){#define AUTOPAINT_UNDO	if (!mActiveTerrain)      return;   S32 mat = getPaintMaterialIndex();   if (mat == -1)      return;	  #ifndef AUTOPAINT_UNDO	  mUndoSel = new Selection;	  #endif   U32 terrBlocks = mActiveTerrain->getBlockSize();   for (U32 y = 0; y < terrBlocks; y++)   {      for (U32 x = 0; x < terrBlocks; x++)      {         // get info         GridPoint gp;         gp.terrainBlock = mActiveTerrain;         gp.gridPos.set(x, y);         GridInfo gi;         getGridInfo(gp, gi);         if (gi.mMaterial == mat)            continue;         if (mRandI(0, 100) > mCoverage)            continue;         Point3F wp;         gridToWorld(gp, wp);         if (!(wp.z >= mMinHeight && wp.z <= mMaxHeight))            continue;         // transform wp to object space         Point3F op;         mActiveTerrain->getWorldTransform().mulP(wp, &op);         Point3F norm;         mActiveTerrain->getNormal(Point2F(op.x, op.y), &norm, true);         if (mMinSlope > 0)            if (norm.z > mSin(mDegToRad(90.0f - mMinSlope)))               continue;         if (mMaxSlope < 90)            if (norm.z < mSin(mDegToRad(90.0f - mMaxSlope)))               continue;         gi.mMaterialChanged = true;         #ifndef AUTOPAINT_UNDO         mUndoSel->add(gi);         #endif         gi.mMaterial = mat;         setGridInfo(gi);      }   }   #ifndef AUTOPAINT_UNDO   if(mUndoSel->size())      submitUndo( mUndoSel );   else      delete mUndoSel;   mUndoSel = 0;   #endif   scheduleMaterialUpdate();}DefineEngineMethod( TerrainEditor, autoMaterialLayer, void, (F32 minHeight, F32 maxHeight, F32 minSlope, F32 maxSlope, F32 coverage),,   "Rule based terrain painting.\n"   "@param minHeight Minimum terrain height."   "@param maxHeight Maximum terrain height."   "@param minSlope Minimum terrain slope."   "@param maxSlope Maximum terrain slope."   "@param coverage Terrain coverage amount."){   object->autoMaterialLayer( minHeight,maxHeight, minSlope, maxSlope, coverage );}
 |