TransformGizmo.cpp 55 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459
  1. /*
  2. Copyright (C) 2013 by Ivan Safrin
  3. Permission is hereby granted, free of charge, to any person obtaining a copy
  4. of this software and associated documentation files (the "Software"), to deal
  5. in the Software without restriction, including without limitation the rights
  6. to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  7. copies of the Software, and to permit persons to whom the Software is
  8. furnished to do so, subject to the following conditions:
  9. The above copyright notice and this permission notice shall be included in
  10. all copies or substantial portions of the Software.
  11. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  12. IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  13. FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  14. AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  15. LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  16. OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  17. THE SOFTWARE.
  18. */
  19. #include "TransformGizmo.h"
  20. #include <cmath>
  21. extern UIGlobalMenu *globalMenu;
  22. TransformGrips::TransformGrips() : UIElement() {
  23. mainRect = new UIRect(1.0, 1.0);
  24. mainRect->setBlendingMode(Renderer::BLEND_MODE_NORMAL);
  25. mainRect->color.setColor(0.0, 0.5, 1.0, 0.2);
  26. grips.push_back(mainRect);
  27. transformTL = new UIImage("spriteEditor/transform_corner.png", 8, 8);
  28. grips.push_back(transformTL);
  29. transformT = new UIImage("spriteEditor/transform_corner.png", 8, 8);
  30. grips.push_back(transformT);
  31. transformTR = new UIImage("spriteEditor/transform_corner.png", 8, 8);
  32. grips.push_back(transformTR);
  33. transformL = new UIImage("spriteEditor/transform_corner.png", 8, 8);
  34. grips.push_back(transformL);
  35. transformR = new UIImage("spriteEditor/transform_corner.png", 8, 8);
  36. grips.push_back(transformR);
  37. transformBL = new UIImage("spriteEditor/transform_corner.png", 8, 8);
  38. grips.push_back(transformBL);
  39. transformB = new UIImage("spriteEditor/transform_corner.png", 8, 8);
  40. grips.push_back(transformB);
  41. transformBR = new UIImage("spriteEditor/transform_corner.png", 8, 8);
  42. grips.push_back(transformBR);
  43. transformOffset = new UIImage("spriteEditor/transform_offset.png", 12, 12);
  44. grips.push_back(transformOffset);
  45. for(int i=0; i < grips.size(); i++) {
  46. addChild(grips[i]);
  47. if(grips[i] != mainRect) {
  48. grips[i]->setAnchorPoint(Vector3());
  49. }
  50. grips[i]->addEventListener(this, InputEvent::EVENT_MOUSEDOWN);
  51. grips[i]->addEventListener(this, InputEvent::EVENT_MOUSEUP);
  52. grips[i]->addEventListener(this, InputEvent::EVENT_MOUSEUP_OUTSIDE);
  53. grips[i]->blockMouseInput = true;
  54. }
  55. Services()->getCore()->getInput()->addEventListener(this, InputEvent::EVENT_MOUSEMOVE);
  56. transforming = false;
  57. movingTransform = NULL;
  58. }
  59. Polycode::Rectangle TransformGrips::getGripRectangle() {
  60. return gripRectangle;
  61. }
  62. void TransformGrips::handleEvent(Event *event) {
  63. if(event->getDispatcher() == Services()->getCore()->getInput()) {
  64. if(transforming) {
  65. Vector2 newMouse = Services()->getCore()->getInput()->getMousePosition();
  66. if(movingTransform == mainRect) {
  67. gripRectangle.x += newMouse.x - mouseBase.x;
  68. gripRectangle.y += newMouse.y - mouseBase.y;
  69. } else if(movingTransform == transformTL) {
  70. gripRectangle.x += newMouse.x - mouseBase.x;
  71. gripRectangle.y += newMouse.y - mouseBase.y;
  72. gripRectangle.w -= newMouse.x - mouseBase.x;
  73. gripRectangle.h -= newMouse.y - mouseBase.y;
  74. } else if(movingTransform == transformTR) {
  75. gripRectangle.y += newMouse.y - mouseBase.y;
  76. gripRectangle.w += newMouse.x - mouseBase.x;
  77. gripRectangle.h -= newMouse.y - mouseBase.y;
  78. } else if(movingTransform == transformT) {
  79. gripRectangle.y += newMouse.y - mouseBase.y;
  80. gripRectangle.h -= newMouse.y - mouseBase.y;
  81. } else if(movingTransform == transformL) {
  82. gripRectangle.x += newMouse.x - mouseBase.x;
  83. gripRectangle.w -= newMouse.x - mouseBase.x;
  84. } else if(movingTransform == transformR) {
  85. gripRectangle.w += newMouse.x - mouseBase.x;
  86. } else if(movingTransform == transformBL) {
  87. gripRectangle.x += newMouse.x - mouseBase.x;
  88. gripRectangle.w -= newMouse.x - mouseBase.x;
  89. gripRectangle.h += newMouse.y - mouseBase.y;
  90. } else if(movingTransform == transformBR) {
  91. gripRectangle.w += newMouse.x - mouseBase.x;
  92. gripRectangle.h += newMouse.y - mouseBase.y;
  93. } else if(movingTransform == transformB) {
  94. gripRectangle.h += newMouse.y - mouseBase.y;
  95. } else if(movingTransform == transformOffset) {
  96. anchorPoint.x += (newMouse.x - mouseBase.x) / gripRectangle.w;
  97. anchorPoint.y += (newMouse.y - mouseBase.y) / gripRectangle.h;
  98. }
  99. mouseBase = newMouse;
  100. dispatchEvent(new Event(), Event::CHANGE_EVENT);
  101. }
  102. } else {
  103. if(event->getEventCode() == InputEvent::EVENT_MOUSEDOWN) {
  104. movingTransform = (UIImage*) event->getDispatcher();
  105. transforming = true;
  106. mouseBase = Services()->getCore()->getInput()->getMousePosition();
  107. } else {
  108. transforming = false;
  109. }
  110. }
  111. }
  112. Vector2 TransformGrips::getAnchorPoint() {
  113. return anchorPoint;
  114. }
  115. TransformGrips::~TransformGrips() {
  116. Services()->getCore()->getInput()->removeAllHandlersForListener(this);
  117. }
  118. void TransformGrips::setGripRectangle(Polycode::Rectangle rectangle, Vector2 offset) {
  119. mainRect->setPosition(rectangle.x, rectangle.y);
  120. mainRect->Resize(rectangle.w, rectangle.h);
  121. transformTL->setPosition(rectangle.x, rectangle.y);
  122. transformT->setPosition(rectangle.x + (rectangle.w * 0.5), rectangle.y);
  123. transformTR->setPosition(rectangle.x + (rectangle.w), rectangle.y);
  124. transformL->setPosition(rectangle.x, rectangle.y +(rectangle.h * 0.5));
  125. transformR->setPosition(rectangle.x + (rectangle.w), rectangle.y+(rectangle.h * 0.5));
  126. transformBL->setPosition(rectangle.x, rectangle.y+rectangle.h);
  127. transformB->setPosition(rectangle.x + (rectangle.w * 0.5), rectangle.y+rectangle.h);
  128. transformBR->setPosition(rectangle.x+rectangle.w, rectangle.y+rectangle.h);
  129. transformOffset->setPosition(rectangle.x + (rectangle.w * 0.5) + (offset.x * rectangle.w), rectangle.y + (rectangle.h * 0.5) + (offset.y * rectangle.h));
  130. gripRectangle = rectangle;
  131. anchorPoint = offset;
  132. }
  133. TrasnformGizmoEvent::TrasnformGizmoEvent(int mode) : Event() {
  134. this->mode = mode;
  135. this->eventCode = eventCode;
  136. eventType = "TrasnformGizmoEvent";
  137. }
  138. TransformGizmoMenu::TransformGizmoMenu(TransformGizmo *gizmo) : UIElement() {
  139. processInputEvents = true;
  140. this->gizmo = gizmo;
  141. transformSelector = new UIIconSelector();
  142. addChild(transformSelector);
  143. transformSelector->addIcon("entityEditor/move_gizmo.png");
  144. transformSelector->addIcon("entityEditor/scale_gizmo.png");
  145. transformSelector->addIcon("entityEditor/rotate_gizmo.png");
  146. transformSelector->setPosition(4, 3.0);
  147. transformSelector->addEventListener(this, UIEvent::SELECT_EVENT);
  148. orientationCombo = new UIComboBox(globalMenu, 100);
  149. orientationCombo->addComboItem("Global");
  150. orientationCombo->addComboItem("Local");
  151. orientationCombo->setSelectedIndex(0);
  152. addChild(orientationCombo);
  153. orientationCombo->setPosition(100, 2);
  154. orientationCombo->addEventListener(this, UIEvent::CHANGE_EVENT);
  155. centerSelector = new UIIconSelector();
  156. addChild(centerSelector);
  157. centerSelector->addIcon("entityEditor/median_center.png");
  158. centerSelector->addIcon("entityEditor/individual_centers.png");
  159. centerSelector->setPosition(210, 3.0);
  160. centerSelector->addEventListener(this, UIEvent::SELECT_EVENT);
  161. }
  162. void TransformGizmoMenu::handleEvent(Event *event) {
  163. if(event->getDispatcher() == transformSelector) {
  164. switch(transformSelector->getSelectedIndex()) {
  165. case 0:
  166. gizmo->setTransformMode(TransformGizmo::TRANSFORM_MOVE);
  167. break;
  168. case 1:
  169. gizmo->setTransformMode(TransformGizmo::TRANSFORM_SCALE);
  170. break;
  171. case 2:
  172. gizmo->setTransformMode(TransformGizmo::TRANSFORM_ROTATE);
  173. break;
  174. }
  175. } else if(event->getDispatcher() == orientationCombo) {
  176. gizmo->setTransformOrientation(orientationCombo->getSelectedIndex());
  177. } else if(event->getDispatcher() == centerSelector) {
  178. if(centerSelector->getSelectedIndex() == 0) {
  179. gizmo->setCenterMode(TransformGizmo::CENTER_MODE_MEDIAN);
  180. } else {
  181. gizmo->setCenterMode(TransformGizmo::CENTER_MODE_INDIVIDUAL);
  182. }
  183. }
  184. }
  185. TransformGizmoMenu::~TransformGizmoMenu() {
  186. }
  187. void TransformGizmo::setCenterMode(int centerMode) {
  188. this->centerMode = centerMode;
  189. }
  190. void TransformGizmo::toggleOrientation() {
  191. if(startingOrientation == -1)
  192. startingOrientation = orientation;
  193. if(orientation == ORIENTATION_GLOBAL)
  194. orientation = ORIENTATION_LOCAL;
  195. else if(orientation == ORIENTATION_LOCAL)
  196. orientation = ORIENTATION_GLOBAL;
  197. }
  198. TransformGizmo::TransformGizmo(Scene *targetScene, Camera *targetCamera) : Entity() {
  199. processInputEvents = true;
  200. orientation = ORIENTATION_GLOBAL;
  201. startingOrientation = -1;
  202. centerMode = CENTER_MODE_MEDIAN;
  203. enableGizmo = true;
  204. firstMove = true;
  205. snapEnabled = false;
  206. snapSize = 1.0;
  207. this->targetScene = targetScene;
  208. this->targetCamera = targetCamera;
  209. ScenePrimitive *centerCircle = new ScenePrimitive(ScenePrimitive::TYPE_LINE_CIRCLE, 0.3, 0.3, 16);
  210. centerCircle->getMesh()->setMeshType(Mesh::LINE_LOOP_MESH);
  211. centerCircle->setColor(0.7, 0.7, 0.7, 1.0);
  212. centerCircle->depthTest = false;
  213. centerCircle->billboardMode = true;
  214. addChild(centerCircle);
  215. centerCircle->setLineWidth(CoreServices::getInstance()->getRenderer()->getBackingResolutionScaleX());
  216. trasnformDecorators = new Entity();
  217. addChild(trasnformDecorators);
  218. scaleDecorators = new Entity();
  219. addChild(scaleDecorators);
  220. transformAndScaleLines = new Entity();
  221. addChild(transformAndScaleLines);
  222. rotateDectorators = new Entity();
  223. addChild(rotateDectorators);
  224. yLine = new SceneMesh(Mesh::LINE_MESH);
  225. yLine->getMesh()->addVertex(0.0, 0.0, 0.0);
  226. yLine->getMesh()->addVertex(0.0, 1.0, 0.0);
  227. yLine->depthTest = false;
  228. yLine->setColor(0.0, 1.0, 0.0, 1.0);
  229. yLine->setLocalBoundingBox(yLine->getMesh()->calculateBBox());
  230. yLine->setLineWidth(CoreServices::getInstance()->getRenderer()->getBackingResolutionScaleX() * 2.0);
  231. transformAndScaleLines->addChild(yLine);
  232. xLine = new SceneMesh(Mesh::LINE_MESH);
  233. xLine->getMesh()->addVertex(0.0, 0.0, 0.0);
  234. xLine->getMesh()->addVertex(1.0, 0.0, 0.0);
  235. xLine->depthTest = false;
  236. xLine->setColor(1.0, 0.0, 0.0, 1.0);
  237. xLine->setLocalBoundingBox(xLine->getMesh()->calculateBBox());
  238. xLine->setLineWidth(CoreServices::getInstance()->getRenderer()->getBackingResolutionScaleX() * 2.0);
  239. transformAndScaleLines->addChild(xLine);
  240. zLine = new SceneMesh(Mesh::LINE_MESH);
  241. zLine->getMesh()->addVertex(0.0, 0.0, 0.0);
  242. zLine->getMesh()->addVertex(0.0, 0.0, 1.0);
  243. zLine->depthTest = false;
  244. zLine->setColor(0.0, 0.0, 1.0, 1.0);
  245. zLine->setLocalBoundingBox(zLine->getMesh()->calculateBBox());
  246. zLine->setLineWidth(CoreServices::getInstance()->getRenderer()->getBackingResolutionScaleX() * 2.0);
  247. transformAndScaleLines->addChild(zLine);
  248. // MOVE
  249. yArrow = new ScenePrimitive(ScenePrimitive::TYPE_CONE, 0.2, 0.05, 12);
  250. yArrow->setColor(0.0, 1.0, 0.0, 1.0);
  251. yArrow->setPosition(0.0, 1.0, 0.0);
  252. yArrow->depthTest = false;
  253. trasnformDecorators->addChild(yArrow);
  254. xArrow = new ScenePrimitive(ScenePrimitive::TYPE_CONE, 0.2, 0.05, 12);
  255. xArrow->setColor(1.0, 0.0, 0.0, 1.0);
  256. xArrow->setPosition(1.0, 0.0, 0.0);
  257. xArrow->Roll(-90);
  258. xArrow->depthTest = false;
  259. trasnformDecorators->addChild(xArrow);
  260. zArrow = new ScenePrimitive(ScenePrimitive::TYPE_CONE, 0.2, 0.05, 12);
  261. zArrow->setColor(0.0, 0.0, 1.0, 1.0);
  262. zArrow->setPosition(0.0, 0.0, 1.0);
  263. zArrow->Pitch(90);
  264. zArrow->depthTest = false;
  265. trasnformDecorators->addChild(zArrow);
  266. // SCALE
  267. yBox = new ScenePrimitive(ScenePrimitive::TYPE_BOX, 0.1, 0.1, 0.1);
  268. yBox->setColor(0.0, 1.0, 0.0, 1.0);
  269. yBox->setPosition(0.0, 1.0, 0.0);
  270. yBox->depthTest = false;
  271. scaleDecorators->addChild(yBox);
  272. xBox = new ScenePrimitive(ScenePrimitive::TYPE_BOX, 0.1, 0.1, 0.1);
  273. xBox->setColor(1.0, 0.0, 0.0, 1.0);
  274. xBox->setPosition(1.0, 0.0, 0.0);
  275. xBox->Roll(-90);
  276. xBox->depthTest = false;
  277. scaleDecorators->addChild(xBox);
  278. zBox = new ScenePrimitive(ScenePrimitive::TYPE_BOX, 0.1, 0.1, 0.1);
  279. zBox->setColor(0.0, 0.0, 1.0, 1.0);
  280. zBox->setPosition(0.0, 0.0, 1.0);
  281. zBox->Pitch(90);
  282. zBox->depthTest = false;
  283. scaleDecorators->addChild(zBox);
  284. // ROTATE
  285. bgCircle = new ScenePrimitive(ScenePrimitive::TYPE_LINE_CIRCLE, 2.6, 2.6, 32);
  286. bgCircle->getMesh()->setMeshType(Mesh::LINE_LOOP_MESH);
  287. bgCircle->setColor(0.0, 0.0, 0.0, 1.0);
  288. bgCircle->depthTest = false;
  289. bgCircle->billboardMode = true;
  290. rotateDectorators->addChild(bgCircle);
  291. bgCircle->setLineWidth(CoreServices::getInstance()->getRenderer()->getBackingResolutionScaleX());
  292. outerCircle = new ScenePrimitive(ScenePrimitive::TYPE_LINE_CIRCLE, 3.0, 3.0, 32);
  293. outerCircle->getMesh()->setMeshType(Mesh::LINE_LOOP_MESH);
  294. outerCircle->setColor(1.0, 1.0, 1.0, 1.0);
  295. outerCircle->depthTest = false;
  296. outerCircle->billboardMode = true;
  297. rotateDectorators->addChild(outerCircle);
  298. outerCircle->setLineWidth(CoreServices::getInstance()->getRenderer()->getBackingResolutionScaleX() * 2.0);
  299. pitchCircle = new ScenePrimitive(ScenePrimitive::TYPE_LINE_CIRCLE, 2.55, 2.55, 32);
  300. pitchCircle->getMesh()->setMeshType(Mesh::LINE_LOOP_MESH);
  301. pitchCircle->setColor(1.0, 0.0, 0.0, 1.0);
  302. pitchCircle->depthTest = false;
  303. pitchCircle->Yaw(90);
  304. rotateDectorators->addChild(pitchCircle);
  305. pitchCircle->setMaterialByName("OneSidedLine");
  306. pitchCircle->setLineWidth(CoreServices::getInstance()->getRenderer()->getBackingResolutionScaleX() * 2.0);
  307. yawCircle = new ScenePrimitive(ScenePrimitive::TYPE_LINE_CIRCLE, 2.65, 2.65, 32);
  308. yawCircle->getMesh()->setMeshType(Mesh::LINE_LOOP_MESH);
  309. yawCircle->setColor(0.0, 1.0, 0.0, 1.0);
  310. yawCircle->depthTest = false;
  311. yawCircle->Pitch(90);
  312. rotateDectorators->addChild(yawCircle);
  313. yawCircle->setMaterialByName("OneSidedLine");
  314. yawCircle->setLineWidth(CoreServices::getInstance()->getRenderer()->getBackingResolutionScaleX()* 2.0);
  315. rollCircle = new ScenePrimitive(ScenePrimitive::TYPE_LINE_CIRCLE, 2.6, 2.6, 32);
  316. rollCircle->getMesh()->setMeshType(Mesh::LINE_LOOP_MESH);
  317. rollCircle->setColor(0.0, 0.0, 1.0, 1.0);
  318. rollCircle->depthTest = false;
  319. rotateDectorators->addChild(rollCircle);
  320. rollCircle->setMaterialByName("OneSidedLine");
  321. rollCircle->setLineWidth(CoreServices::getInstance()->getRenderer()->getBackingResolutionScaleX()* 2.0);
  322. rotateDectorators->processInputEvents = true;
  323. //pitchGrip = new ScenePrimitive(ScenePrimitive::TYPE_TORUS, 1.55 * 0.5, 0.05, 16, 3);
  324. pitchGrip = new ScenePrimitive(ScenePrimitive::TYPE_UNCAPPED_CYLINDER, 0.15, 2.55 * 0.5, 16);
  325. pitchGrip->setColor(1.0, 0.0, 0.0, 0.2);
  326. pitchGrip->depthTest = false;
  327. pitchGrip->Pitch(90);
  328. pitchGrip->Yaw(90);
  329. rotateDectorators->addChild(pitchGrip);
  330. pitchGrip->processInputEvents = true;
  331. pitchGrip->addEventListener(this, InputEvent::EVENT_MOUSEDOWN);
  332. pitchGrip->useGeometryHitDetection = true;
  333. pitchGrip->blockMouseInput = true;
  334. rollGrip = new ScenePrimitive(ScenePrimitive::TYPE_UNCAPPED_CYLINDER, 0.15, 2.6 * 0.5, 16);
  335. rollGrip->setColor(0.0, 0.0, 1.0, 0.2);
  336. rollGrip->depthTest = false;
  337. rollGrip->Pitch(90);
  338. rotateDectorators->addChild(rollGrip);
  339. rollGrip->processInputEvents = true;
  340. rollGrip->addEventListener(this, InputEvent::EVENT_MOUSEDOWN);
  341. rollGrip->useGeometryHitDetection = true;
  342. rollGrip->blockMouseInput = true;
  343. yawGrip= new ScenePrimitive(ScenePrimitive::TYPE_UNCAPPED_CYLINDER, 0.15, 2.65 * 0.5, 16);
  344. yawGrip->setColor(0.0, 1.0, 0.0, 0.2);
  345. yawGrip->depthTest = false;
  346. yawGrip->Yaw(90);
  347. rotateDectorators->addChild(yawGrip);
  348. yawGrip->processInputEvents = true;
  349. yawGrip->addEventListener(this, InputEvent::EVENT_MOUSEDOWN);
  350. yawGrip->useGeometryHitDetection = true;
  351. yawGrip->blockMouseInput = true;
  352. viewportRotateGripBase = new Entity();
  353. viewportRotateGripBase->processInputEvents = true;
  354. rotateDectorators->addChild(viewportRotateGripBase);
  355. viewportRotateGrip = new ScenePrimitive(ScenePrimitive::TYPE_TORUS, 3.0 * 0.5, 0.08, 16, 3);
  356. viewportRotateGrip->Pitch(90);
  357. viewportRotateGrip->setColor(0.0, 1.0, 0.0, 0.2);
  358. viewportRotateGrip->depthTest = false;
  359. viewportRotateGripBase->addChild(viewportRotateGrip);
  360. viewportRotateGrip->processInputEvents = true;
  361. viewportRotateGrip->addEventListener(this, InputEvent::EVENT_MOUSEDOWN);
  362. viewportRotateGrip->useGeometryHitDetection = true;
  363. viewportRotateGrip->blockMouseInput = true;
  364. pitchGrip->visible = false;
  365. yawGrip->visible = false;
  366. rollGrip->visible = false;
  367. viewportRotateGrip->visible = false;
  368. xTransformGrip = new Entity();
  369. xTransformGrip->setLocalBoundingBox(1.3, 0.1, 0.1);
  370. xTransformGrip->depthTest = false;
  371. xTransformGrip->setColor(1.0, 0.0, 0.0, 1.0);
  372. addChild(xTransformGrip);
  373. xTransformGrip->setAnchorPoint(Vector3(-1.0, 0.0, 0.0));
  374. xTransformGrip->processInputEvents = true;
  375. xTransformGrip->addEventListener(this, InputEvent::EVENT_MOUSEDOWN);
  376. yTransformGrip = new Entity();
  377. yTransformGrip->setLocalBoundingBox(0.1, 1.3, 0.1);
  378. addChild(yTransformGrip);
  379. yTransformGrip->setAnchorPoint(Vector3(0.0, -1.0, 0.0));
  380. yTransformGrip->processInputEvents = true;
  381. yTransformGrip->addEventListener(this, InputEvent::EVENT_MOUSEDOWN);
  382. zTransformGrip = new Entity();
  383. zTransformGrip->setLocalBoundingBox(0.1, 0.1, 1.3);
  384. addChild(zTransformGrip);
  385. zTransformGrip->setAnchorPoint(Vector3(0.0, 0.0, -1.0));
  386. zTransformGrip->processInputEvents = true;
  387. zTransformGrip->addEventListener(this, InputEvent::EVENT_MOUSEDOWN);
  388. transforming = false;
  389. mode = TRANSFORM_MOVE;
  390. visible = false;
  391. enabled = false;
  392. coreInput = CoreServices::getInstance()->getCore()->getInput();
  393. coreInput->addEventListener(this, InputEvent::EVENT_MOUSEMOVE);
  394. coreInput->addEventListener(this, InputEvent::EVENT_MOUSEUP);
  395. coreInput->addEventListener(this, InputEvent::EVENT_MOUSEDOWN);
  396. coreInput->addEventListener(this, InputEvent::EVENT_KEYDOWN);
  397. gizmoMode = GIZMO_MODE_3D;
  398. setTransformMode(TRANSFORM_MOVE);
  399. }
  400. void TransformGizmo::enableSnap(bool val) {
  401. snapEnabled = val;
  402. }
  403. void TransformGizmo::setSnapSize(Number snapSize) {
  404. this->snapSize = snapSize;
  405. }
  406. void TransformGizmo::setTransformOrientation(int orientation) {
  407. this->orientation = orientation;
  408. }
  409. void TransformGizmo::setTransformPlaneFromView() {
  410. switch(gizmoMode) {
  411. case GIZMO_MODE_2D_X:
  412. setTransformPlane(1.0, 0.0, 0.0, true);
  413. transformConstraint = Vector3(0.0, 1.0, 1.0);
  414. break;
  415. case GIZMO_MODE_2D_Y:
  416. setTransformPlane(0.0, 1.0, 0.0, true);
  417. transformConstraint = Vector3(1.0, 0.0, 1.0);
  418. break;
  419. case GIZMO_MODE_2D_Z:
  420. setTransformPlane(0.0, 0.0, 1.0, true);
  421. transformConstraint = Vector3(1.0, 1.0, 0.0);
  422. break;
  423. default:
  424. Vector3 camVec = targetCamera->getConcatenatedMatrix().getPosition() - getConcatenatedMatrix().getPosition();
  425. camVec.Normalize();
  426. setTransformPlane(camVec.x, camVec.y, camVec.z, true);
  427. break;
  428. }
  429. }
  430. void TransformGizmo::setTransformPlane(Number x, Number y, Number z, bool forceGlobal) {
  431. planeMatrix = getConcatenatedMatrix();
  432. Vector3 localPlane = Vector3(x,y,z);
  433. localTransformPlane = localPlane;
  434. Number planeDistance = 0;
  435. if(forceGlobal) {
  436. transformPlane = localPlane;
  437. transformPlaneDistance = planeDistance;
  438. planeMatrix.identity();
  439. return;
  440. }
  441. transformPlane = planeMatrix.rotateVector(localPlane);
  442. Vector3 planePoint = planeMatrix.getPosition();
  443. transformPlaneDistance = planePoint.dot(transformPlane);
  444. Ray gizmoRay;
  445. gizmoRay.origin = 0.0;
  446. gizmoRay.direction = transformPlane * -1;
  447. gizmoPoint = gizmoRay.planeIntersectPoint(transformPlane, transformPlaneDistance);
  448. gizmoPoint = planeMatrix.Inverse() * gizmoPoint;
  449. }
  450. void TransformGizmo::setTransformPlane(bool useX, bool useY, bool useZ) {
  451. Vector3 xDir = getAnchorAdjustedMatrix().rotateVector(Vector3(1.0, 0.0, 0.0));
  452. Vector3 yDir = getAnchorAdjustedMatrix().rotateVector(Vector3(0.0, 1.0, 0.0));
  453. Vector3 zDir = getAnchorAdjustedMatrix().rotateVector(Vector3(0.0, 0.0, 1.0));
  454. Number xDot = abs(targetCamera->getRotationQuat().applyTo(Vector3(0.0, 0.0, -1.0)).dot(xDir));
  455. Number yDot = abs(targetCamera->getRotationQuat().applyTo(Vector3(0.0, 0.0, -1.0)).dot(yDir));
  456. Number zDot = abs(targetCamera->getRotationQuat().applyTo(Vector3(0.0, 0.0, -1.0)).dot(zDir));
  457. if(useX && !useY && !useZ) {
  458. if(yDot > zDot)
  459. setTransformPlane(0.0, 1.0, 0.0);
  460. else
  461. setTransformPlane(0.0, 0.0, 1.0);
  462. } else if(!useX && useY && !useZ) {
  463. if(xDot > zDot)
  464. setTransformPlane(1.0, 0.0, 0.0);
  465. else
  466. setTransformPlane(0.0, 0.0, 1.0);
  467. } else if(!useX && !useY && useZ) {
  468. if(xDot > yDot)
  469. setTransformPlane(1.0, 0.0, 0.0);
  470. else
  471. setTransformPlane(0.0, 1.0, 0.0);
  472. } else if(!useX) {
  473. setTransformPlane(1.0, 0.0, 0.0);
  474. } else if(!useY) {
  475. setTransformPlane(0.0, 1.0, 0.0);
  476. } else if(!useZ) {
  477. setTransformPlane(0.0, 0.0, 1.0);
  478. }
  479. }
  480. Vector3 TransformGizmo::getTransformPlanePosition() {
  481. Ray ray = targetScene->projectRayFromCameraAndViewportCoordinate(targetCamera, coreInput->getMousePosition());
  482. // hack to fix NaN's in the plane intersect math (sorry)
  483. ray.direction += Vector3(0.00001, 0.00001, 0.00001);
  484. Vector3 ret = ray.planeIntersectPoint(transformPlane, transformPlaneDistance);
  485. return ret;
  486. }
  487. Vector3 TransformGizmo::getPositionAlongAxis() {
  488. Ray ray = Ray(position, rotationQuat.applyTo(transformConstraint));
  489. Ray ray2 = targetScene->projectRayFromCameraAndViewportCoordinate(targetCamera, coreInput->getMousePosition());
  490. Vector3 ret;
  491. ray.closestPointsBetween(ray2, &ret, NULL);
  492. return ret;
  493. }
  494. bool TransformGizmo::isConstrainedToSingleAxis() {
  495. if(transformConstraint.x == 1.0) {
  496. if(transformConstraint.y == 0.0 && transformConstraint.z == 0.0)
  497. return true;
  498. }
  499. if(transformConstraint.y == 1.0) {
  500. if(transformConstraint.x == 0.0 && transformConstraint.z == 0.0)
  501. return true;
  502. }
  503. if(transformConstraint.z == 1.0) {
  504. if(transformConstraint.x == 0.0 && transformConstraint.y == 0.0)
  505. return true;
  506. }
  507. return false;
  508. }
  509. void TransformGizmo::setTransformMode(int newMode) {
  510. transformMode = newMode;
  511. trasnformDecorators->visible = false;
  512. scaleDecorators->visible = false;
  513. transformAndScaleLines->visible = false;
  514. rotateDectorators->visible = false;
  515. xTransformGrip->enabled = false;
  516. yTransformGrip->enabled = false;
  517. zTransformGrip->enabled = false;
  518. pitchGrip->enabled = false;
  519. rollGrip->enabled = false;
  520. yawGrip->enabled = false;
  521. viewportRotateGrip->enabled = false;
  522. xTransformGrip->enabled = false;
  523. yTransformGrip->enabled = false;
  524. zTransformGrip->enabled = false;
  525. mode = newMode;
  526. switch (mode) {
  527. case TRANSFORM_MOVE:
  528. trasnformDecorators->visible = true;
  529. transformAndScaleLines->visible = true;
  530. switch(gizmoMode) {
  531. case GIZMO_MODE_3D:
  532. xTransformGrip->enabled = true;
  533. yTransformGrip->enabled = true;
  534. zTransformGrip->enabled = true;
  535. xArrow->visible = true;
  536. yArrow->visible = true;
  537. zArrow->visible = true;
  538. xLine->visible = true;
  539. yLine->visible = true;
  540. zLine->visible = true;
  541. break;
  542. case GIZMO_MODE_2D_X:
  543. xTransformGrip->enabled = false;
  544. yTransformGrip->enabled = true;
  545. zTransformGrip->enabled = true;
  546. xArrow->visible = false;
  547. yArrow->visible = true;
  548. zArrow->visible = true;
  549. xLine->visible = false;
  550. yLine->visible = true;
  551. zLine->visible = true;
  552. break;
  553. case GIZMO_MODE_2D_Y:
  554. xTransformGrip->enabled = true;
  555. yTransformGrip->enabled = false;
  556. zTransformGrip->enabled = true;
  557. xArrow->visible = true;
  558. yArrow->visible = false;
  559. zArrow->visible = true;
  560. xLine->visible = true;
  561. yLine->visible = false;
  562. zLine->visible = true;
  563. break;
  564. case GIZMO_MODE_2D_Z:
  565. xTransformGrip->enabled = true;
  566. yTransformGrip->enabled = true;
  567. zTransformGrip->enabled = false;
  568. xArrow->visible = true;
  569. yArrow->visible = true;
  570. zArrow->visible = false;
  571. xLine->visible = true;
  572. yLine->visible = true;
  573. zLine->visible = false;
  574. break;
  575. }
  576. break;
  577. case TRANSFORM_SCALE:
  578. scaleDecorators->visible = true;
  579. transformAndScaleLines->visible = true;
  580. switch(gizmoMode) {
  581. case GIZMO_MODE_3D:
  582. xTransformGrip->enabled = true;
  583. yTransformGrip->enabled = true;
  584. zTransformGrip->enabled = true;
  585. xBox->visible = true;
  586. yBox->visible = true;
  587. zBox->visible = true;
  588. xLine->visible = true;
  589. yLine->visible = true;
  590. zLine->visible = true;
  591. break;
  592. case GIZMO_MODE_2D_X:
  593. xTransformGrip->enabled = false;
  594. yTransformGrip->enabled = true;
  595. zTransformGrip->enabled = true;
  596. xBox->visible = false;
  597. yBox->visible = true;
  598. zBox->visible = true;
  599. xLine->visible = false;
  600. yLine->visible = true;
  601. zLine->visible = true;
  602. break;
  603. case GIZMO_MODE_2D_Y:
  604. xTransformGrip->enabled = true;
  605. yTransformGrip->enabled = false;
  606. zTransformGrip->enabled = true;
  607. xBox->visible = true;
  608. yBox->visible = false;
  609. zBox->visible = true;
  610. xLine->visible = true;
  611. yLine->visible = false;
  612. zLine->visible = true;
  613. break;
  614. case GIZMO_MODE_2D_Z:
  615. xTransformGrip->enabled = true;
  616. yTransformGrip->enabled = true;
  617. zTransformGrip->enabled = false;
  618. xBox->visible = true;
  619. yBox->visible = true;
  620. zBox->visible = false;
  621. xLine->visible = true;
  622. yLine->visible = true;
  623. zLine->visible = false;
  624. break;
  625. }
  626. break;
  627. case TRANSFORM_ROTATE:
  628. rotateDectorators->visible = true;
  629. switch(gizmoMode) {
  630. case GIZMO_MODE_3D:
  631. rollGrip->enabled = true;
  632. rollCircle->visible = true;
  633. pitchGrip->enabled = true;
  634. pitchCircle->visible = true;
  635. yawGrip->enabled = true;
  636. yawCircle->visible = true;
  637. viewportRotateGrip->enabled = false;
  638. outerCircle->visible = false;
  639. outerCircle->setColor(1.0, 1.0, 1.0, 1.0);
  640. break;
  641. case GIZMO_MODE_2D_X:
  642. rollGrip->enabled = false;
  643. rollCircle->visible = false;
  644. pitchGrip->enabled = false;
  645. pitchCircle->visible = false;
  646. yawGrip->enabled = false;
  647. yawCircle->visible = false;
  648. viewportRotateGrip->enabled = true;
  649. outerCircle->visible = true;
  650. outerCircle->setColor(1.0, 0.0, 0.0, 1.0);
  651. viewportRotateGrip->setYaw(90);
  652. viewportRotateGrip->setPitch(90);
  653. break;
  654. case GIZMO_MODE_2D_Y:
  655. rollGrip->enabled = false;
  656. rollCircle->visible = false;
  657. pitchGrip->enabled = false;
  658. pitchCircle->visible = false;
  659. yawGrip->enabled = false;
  660. yawCircle->visible = false;
  661. viewportRotateGrip->enabled = true;
  662. outerCircle->visible = true;
  663. outerCircle->setColor(0.0, 1.0, 0.0, 1.0);
  664. viewportRotateGrip->setPitch(0);
  665. viewportRotateGrip->setYaw(0);
  666. break;
  667. case GIZMO_MODE_2D_Z:
  668. rollGrip->enabled = false;
  669. rollCircle->visible = false;
  670. pitchGrip->enabled = false;
  671. pitchCircle->visible = false;
  672. yawGrip->enabled = false;
  673. yawCircle->visible = false;
  674. viewportRotateGrip->enabled = true;
  675. outerCircle->visible = true;
  676. outerCircle->setColor(0.0, 0.0, 1.0, 1.0);
  677. viewportRotateGrip->setPitch(90);
  678. viewportRotateGrip->setYaw(0);
  679. break;
  680. }
  681. break;
  682. default:
  683. assert(false); // invalid mode
  684. break;
  685. }
  686. }
  687. void TransformGizmo::setTransformSelection(std::vector<Entity*> selectedEntities) {
  688. entityPositions.clear();
  689. this->selectedEntities = selectedEntities;
  690. if(selectedEntities.size() > 0) {
  691. visible = true;
  692. enabled = true;
  693. Vector3 centerPoint;
  694. for(int i=0; i < selectedEntities.size(); i++) {
  695. centerPoint += selectedEntities[i]->getConcatenatedMatrix().getPosition();
  696. entityPositions.push_back(selectedEntities[i]->getPosition());
  697. }
  698. centerPoint = centerPoint / selectedEntities.size();
  699. setPosition(centerPoint);
  700. } else {
  701. visible = false;
  702. enabled = false;
  703. }
  704. }
  705. void TransformGizmo::resetTransform() {
  706. if(firstMove) return;
  707. for(int i=0; i < selectedEntities.size(); i++) {
  708. selectedEntities[i]->setPosition(oldPosition[i]);
  709. entityPositions[i] = oldPosition[i];
  710. selectedEntities[i]->setScale(oldScale[i]);
  711. selectedEntities[i]->setRotationByQuaternion(oldRotation[i]);
  712. }
  713. if(isConstrainedToSingleAxis())
  714. startingPoint = getPositionAlongAxis();
  715. else
  716. startingPoint = getTransformPlanePosition();
  717. }
  718. void TransformGizmo::transformSelectedEntities(const Vector3 &move, const Vector3 &scale, Number rotate) {
  719. if(firstMove) {
  720. firstMove = false;
  721. dispatchEvent(new TrasnformGizmoEvent(mode), Event::SELECT_EVENT);
  722. oldPosition.clear();
  723. oldScale.clear();
  724. oldRotation.clear();
  725. for(int i=0; i < selectedEntities.size(); i++) {
  726. oldPosition.push_back(selectedEntities[i]->getPosition());
  727. oldScale.push_back(selectedEntities[i]->getScale());
  728. oldRotation.push_back(selectedEntities[i]->getRotationQuat());
  729. }
  730. }
  731. Vector3 globalCenter = getConcatenatedMatrix().getPosition();
  732. for(int i=0; i < selectedEntities.size(); i++) {
  733. if((orientation == ORIENTATION_GLOBAL && mode != TRANSFORM_SCALE_VIEW)) {//|| (ORIENTATION_LOCAL && mode == TRANSFORM_MOVE_VIEW)) {
  734. entityPositions[i] += move;
  735. Quaternion q;
  736. Quaternion currentRotation = selectedEntities[i]->getRotationQuat();
  737. Vector3 axisVector = transformConstraint;
  738. axisVector = currentRotation.Inverse().applyTo(axisVector);
  739. axisVector.Normalize();
  740. q.fromAngleAxis(rotate, axisVector);
  741. Vector3 newScale = selectedEntities[i]->getRotationQuat().applyTo(scale);
  742. newScale.x = fabs(newScale.x);
  743. newScale.y = fabs(newScale.y);
  744. newScale.z = fabs(newScale.z);
  745. if(scale.x < 0 || scale.y < 0 || scale.z < 0) {
  746. newScale = newScale * -1.0;
  747. }
  748. if(centerMode == CENTER_MODE_MEDIAN) {
  749. Vector3 globalPosition = selectedEntities[i]->getConcatenatedMatrix().getPosition();
  750. Quaternion tQ;
  751. tQ.fromAngleAxis(rotate, transformConstraint);
  752. Vector3 trans = globalCenter + tQ.applyTo(globalPosition-globalCenter) - globalPosition;
  753. globalPosition += trans;
  754. selectedEntities[i]->setPosition(globalPosition - selectedEntities[i]->getParentEntity()->getConcatenatedMatrix().getPosition());
  755. selectedEntities[i]->setRotationByQuaternion(currentRotation * q);
  756. if(move.length() == 0.0) {
  757. entityPositions[i] = selectedEntities[i]->getPosition();
  758. }
  759. selectedEntities[i]->setScale(selectedEntities[i]->getScale() * (Vector3(1.0, 1.0, 1.0)+newScale));
  760. if(newScale.length() > 0.0) {
  761. Vector3 scalePosition;
  762. scalePosition.x = globalPosition.x + ((globalPosition.x - globalCenter.x) * newScale.x);
  763. scalePosition.y = globalPosition.y + ((globalPosition.y - globalCenter.y) * newScale.y);
  764. scalePosition.z = globalPosition.z + ((globalPosition.z - globalCenter.z) * newScale.z);
  765. scalePosition = selectedEntities[i]->getParentEntity()->getConcatenatedMatrix().Inverse().transpose() * scalePosition;
  766. selectedEntities[i]->setPosition(scalePosition);
  767. if(move.length() == 0.0) {
  768. entityPositions[i] = selectedEntities[i]->getPosition();
  769. }
  770. }
  771. } else {
  772. selectedEntities[i]->setRotationByQuaternion(currentRotation * q);
  773. if(mode == TRANSFORM_SCALE_VIEW)
  774. selectedEntities[i]->setScale(oldScale[i] * scale);
  775. else
  776. selectedEntities[i]->setScale(selectedEntities[i]->getScale() * (Vector3(1.0, 1.0, 1.0)+newScale));
  777. }
  778. } else {
  779. entityPositions[i] += getRotationQuat().applyTo(move);
  780. Quaternion q;
  781. Quaternion currentRotation = selectedEntities[i]->getRotationQuat();
  782. Vector3 axisVector = transformConstraint;
  783. // always global in the 2d view
  784. if(gizmoMode != GIZMO_MODE_3D) {
  785. axisVector = currentRotation.Inverse().applyTo(axisVector);
  786. }
  787. axisVector.Normalize();
  788. q.fromAngleAxis(rotate, axisVector);
  789. if(centerMode == CENTER_MODE_MEDIAN) {
  790. Vector3 globalPosition = selectedEntities[i]->getConcatenatedMatrix().getPosition();
  791. Quaternion tQ;
  792. tQ.fromAngleAxis(rotate, getRotationQuat().applyTo(axisVector));
  793. Vector3 trans = globalCenter + tQ.applyTo(globalPosition-globalCenter) - globalPosition;
  794. globalPosition += trans;
  795. selectedEntities[i]->setPosition(globalPosition - selectedEntities[i]->getParentEntity()->getConcatenatedMatrix().getPosition());
  796. selectedEntities[i]->setRotationByQuaternion(currentRotation * q);
  797. if(move.length() == 0.0) {
  798. entityPositions[i] = selectedEntities[i]->getPosition();
  799. }
  800. if(mode == TRANSFORM_SCALE_VIEW)
  801. selectedEntities[i]->setScale(oldScale[i] * scale);
  802. else
  803. selectedEntities[i]->setScale(selectedEntities[i]->getScale() * (Vector3(1.0, 1.0, 1.0)+scale));
  804. if(scale.length() > 0.0) {
  805. Vector3 scalePosition;
  806. scalePosition.x = globalPosition.x + ((globalPosition.x - globalCenter.x) * scale.x);
  807. scalePosition.y = globalPosition.y + ((globalPosition.y - globalCenter.y) * scale.y);
  808. scalePosition.z = globalPosition.z + ((globalPosition.z - globalCenter.z) * scale.z);
  809. scalePosition = selectedEntities[i]->getParentEntity()->getConcatenatedMatrix().Inverse().transpose() * scalePosition;
  810. selectedEntities[i]->setPosition(scalePosition);
  811. if(move.length() == 0.0) {
  812. entityPositions[i] = selectedEntities[i]->getPosition();
  813. }
  814. }
  815. } else {
  816. selectedEntities[i]->setRotationByQuaternion(currentRotation * q);
  817. if(mode == TRANSFORM_SCALE_VIEW)
  818. selectedEntities[i]->setScale(oldScale[i] * scale);
  819. else
  820. selectedEntities[i]->setScale(selectedEntities[i]->getScale() * (Vector3(1.0, 1.0, 1.0)+scale));
  821. }
  822. }
  823. // snap if moving and snap is on
  824. if(scale.length() == 0.0 && rotate == 0.0) {
  825. if(snapEnabled) {
  826. Vector3 snappedPositon = entityPositions[i];
  827. snappedPositon.x = round(((Number)snappedPositon.x)/(snapSize)) * snapSize;
  828. snappedPositon.y = round(((Number)snappedPositon.y)/(snapSize)) * snapSize;
  829. snappedPositon.z = round(((Number)snappedPositon.z)/(snapSize)) * snapSize;
  830. selectedEntities[i]->setPosition(snappedPositon);
  831. } else {
  832. selectedEntities[i]->setPosition(entityPositions[i]);
  833. }
  834. }
  835. }
  836. }
  837. Number TransformGizmo::getTransformPlaneAngle() {
  838. Ray gizmoRay;
  839. gizmoRay.origin = getConcatenatedMatrix().getPosition();
  840. gizmoRay.direction = localTransformPlane * -1;
  841. Vector3 gizmoIntersect = gizmoRay.planeIntersectPoint(transformPlane, transformPlaneDistance);
  842. gizmoIntersect = planeMatrix.Inverse() * gizmoIntersect;
  843. Ray ray = targetScene->projectRayFromCameraAndViewportCoordinate(targetCamera, coreInput->getMousePosition());
  844. Vector3 mouseIntersect = ray.planeIntersectPoint(transformPlane, transformPlaneDistance);
  845. mouseIntersect = planeMatrix.Inverse() * mouseIntersect;
  846. Vector2 planePosition;
  847. if(localTransformPlane.x > 0) {
  848. planePosition.x = mouseIntersect.z - gizmoIntersect.z;
  849. planePosition.y = mouseIntersect.y - gizmoIntersect.y;
  850. } else if(localTransformPlane.y > 0.0) {
  851. planePosition.x = mouseIntersect.x - gizmoIntersect.x;
  852. planePosition.y = mouseIntersect.z - gizmoIntersect.z;
  853. } else if(localTransformPlane.z > 0.0) {
  854. planePosition.x = mouseIntersect.x - gizmoIntersect.x;
  855. planePosition.y = mouseIntersect.y - gizmoIntersect.y;
  856. }
  857. planePosition.Normalize();
  858. return atan2(planePosition.x, planePosition.y);
  859. }
  860. Vector2 TransformGizmo::getCorrectedMousePosition() {
  861. Vector2 localMouse = CoreServices::getInstance()->getInput()->getMousePosition();
  862. localMouse.x -= targetScene->sceneMouseRect.x;
  863. localMouse.y -= targetScene->sceneMouseRect.y;
  864. return localMouse;
  865. }
  866. Number TransformGizmo::get2dAngle() {
  867. Polycode::Rectangle view = targetScene->sceneMouseRect;
  868. Polycode::Rectangle camView = targetCamera->getViewport();
  869. Vector2 origin = getScreenPosition(targetCamera->getProjectionMatrix(), targetCamera->getAnchorAdjustedMatrix(), camView);
  870. Vector2 localStart = mouseStart2d - origin;
  871. Vector2 localMouse = CoreServices::getInstance()->getInput()->getMousePosition();
  872. localMouse.x -= view.x;
  873. localMouse.y -= view.y;
  874. localMouse -= origin;
  875. Number ang = atan2(localStart.crossProduct(localMouse), localStart.dot(localMouse));
  876. Number dot = targetCamera->getRotationQuat().applyTo(Vector3(0,0,-1)).dot(rotationQuat.applyTo(transformConstraint));
  877. if(dot < 0.0)
  878. ang *= -1.0;
  879. else {
  880. if(gizmoMode == GIZMO_MODE_2D_X || gizmoMode == GIZMO_MODE_2D_Y)
  881. ang *= -1.0;
  882. }
  883. return ang;
  884. }
  885. void TransformGizmo::setGizmoMode(int newMode) {
  886. gizmoMode = newMode;
  887. setTransformMode(transformMode);
  888. }
  889. void TransformGizmo::handleEvent(Event *event) {
  890. if(!enableGizmo) {
  891. return;
  892. }
  893. if(!coreInput->getKeyState(KEY_LALT) && !coreInput->getKeyState(KEY_RALT)) {
  894. if(event->getDispatcher() == pitchGrip) {
  895. if(event->getEventCode() == InputEvent::EVENT_MOUSEDOWN) {
  896. if(((InputEvent*)event)->getMouseButton() != CoreInput::MOUSE_BUTTON3) {
  897. transforming = true;
  898. transformConstraint = Vector3(1.0, 0.0, 0.0);
  899. setTransformPlane(1.0, 0.0, 0.0);
  900. mouseStart2d = getCorrectedMousePosition();
  901. startingAngle = get2dAngle();
  902. }
  903. }
  904. } else if(event->getDispatcher() == yawGrip) {
  905. if(event->getEventCode() == InputEvent::EVENT_MOUSEDOWN) {
  906. if(((InputEvent*)event)->getMouseButton() != CoreInput::MOUSE_BUTTON3) {
  907. transforming = true;
  908. transformConstraint = Vector3(0.0, 1.0, 0.0);
  909. setTransformPlane(0.0, 1.0, 0.0);
  910. mouseStart2d = getCorrectedMousePosition();
  911. startingAngle = get2dAngle();
  912. }
  913. }
  914. } else if(event->getDispatcher() == rollGrip) {
  915. if(event->getEventCode() == InputEvent::EVENT_MOUSEDOWN) {
  916. if(((InputEvent*)event)->getMouseButton() != CoreInput::MOUSE_BUTTON3) {
  917. transforming = true;
  918. transformConstraint = Vector3(0.0, 0.0, 1.0);
  919. setTransformPlane(0.0, 0.0, -1.0);
  920. mouseStart2d = getCorrectedMousePosition();
  921. startingAngle = get2dAngle();
  922. }
  923. }
  924. } else if(event->getDispatcher() == viewportRotateGrip) {
  925. if(event->getEventCode() == InputEvent::EVENT_MOUSEDOWN) {
  926. if(((InputEvent*)event)->getMouseButton() != CoreInput::MOUSE_BUTTON3) {
  927. if(gizmoMode != GIZMO_MODE_3D) {
  928. transforming = true;
  929. setTransformPlaneFromView();
  930. transformConstraint = transformPlane;
  931. mouseStart2d = getCorrectedMousePosition();
  932. startingAngle = get2dAngle();
  933. }
  934. }
  935. }
  936. }
  937. if(event->getDispatcher() == xTransformGrip) {
  938. if(event->getEventCode() == InputEvent::EVENT_MOUSEDOWN) {
  939. if(((InputEvent*)event)->getMouseButton() != CoreInput::MOUSE_BUTTON3) {
  940. transforming = true;
  941. transformConstraint = Vector3(1.0, 0.0, 0.0);
  942. setTransformPlane(true, false, false);
  943. startingPoint = getTransformPlanePosition();
  944. }
  945. }
  946. } else if(event->getDispatcher() == yTransformGrip) {
  947. if(event->getEventCode() == InputEvent::EVENT_MOUSEDOWN) {
  948. if(((InputEvent*)event)->getMouseButton() != CoreInput::MOUSE_BUTTON3) {
  949. transforming = true;
  950. transformConstraint = Vector3(0.0, 1.0, 0.0);
  951. setTransformPlane(false, true, false);
  952. startingPoint = getTransformPlanePosition();
  953. }
  954. }
  955. } else if(event->getDispatcher() == zTransformGrip) {
  956. if(event->getEventCode() == InputEvent::EVENT_MOUSEDOWN) {
  957. if(((InputEvent*)event)->getMouseButton() != CoreInput::MOUSE_BUTTON3) {
  958. transforming = true;
  959. transformConstraint = Vector3(0.0, 0.0, 1.0);
  960. setTransformPlane(false, false, true);
  961. startingPoint = getTransformPlanePosition();
  962. }
  963. }
  964. }
  965. }
  966. if(event->getDispatcher() == coreInput) {
  967. if (!coreInput->getKeyState(KEY_LCTRL) && !coreInput->getKeyState(KEY_RCTRL) && !coreInput->getKeyState(KEY_LALT) && !coreInput->getKeyState(KEY_RALT)) {
  968. if (event->getEventCode() == InputEvent::EVENT_KEYDOWN) {
  969. InputEvent *inputEvent = (InputEvent*)event;
  970. switch (inputEvent->key) {
  971. case KEY_s:
  972. {
  973. transforming = true;
  974. transformConstraint = Vector3(1.0, 1.0, 1.0);
  975. previousMode = mode;
  976. mode = TRANSFORM_SCALE_VIEW;
  977. setTransformPlaneFromView();
  978. startingPoint = getTransformPlanePosition();
  979. mouseStart2d = getCorrectedMousePosition();
  980. scaleAmount = 0.0;
  981. resetTransform();
  982. }
  983. break;
  984. case KEY_r:
  985. {
  986. previousMode = mode;
  987. mode = TRANSFORM_ROTATE_VIEW;
  988. transforming = true;
  989. transformConstraint = Vector3(1.0, 1.0, 1.0);
  990. setTransformPlaneFromView();
  991. transformConstraint = transformPlane;
  992. mouseStart2d = getCorrectedMousePosition();
  993. startingAngle = get2dAngle();
  994. resetTransform();
  995. }
  996. break;
  997. case KEY_g:
  998. {
  999. previousMode = mode;
  1000. mode = TRANSFORM_MOVE_VIEW;
  1001. transforming = true;
  1002. transformConstraint = Vector3(1.0, 1.0, 1.0);
  1003. setTransformPlaneFromView();
  1004. startingPoint = getTransformPlanePosition();
  1005. resetTransform();
  1006. }
  1007. break;
  1008. case KEY_x:
  1009. {
  1010. if(transforming && (mode == TRANSFORM_MOVE_VIEW || mode == TRANSFORM_SCALE_VIEW || ((mode == TRANSFORM_ROTATE_VIEW) && (gizmoMode == GIZMO_MODE_3D)))) {
  1011. if((coreInput->getKeyState(KEY_LSHIFT) || coreInput->getKeyState(KEY_RSHIFT)) && (mode != TRANSFORM_ROTATE_VIEW) && ((gizmoMode == GIZMO_MODE_2D_X) || (gizmoMode == GIZMO_MODE_3D))) {
  1012. if(transformConstraint != Vector3(0.0, 1.0, 1.0)) {
  1013. transformConstraint = Vector3(0.0, 1.0, 1.0);
  1014. setTransformPlane(false, true, true);
  1015. resetTransform();
  1016. } else {
  1017. toggleOrientation();
  1018. resetTransform();
  1019. }
  1020. } else if(gizmoMode != GIZMO_MODE_2D_X){
  1021. if(transformConstraint != Vector3(1.0, 0.0, 0.0)) {
  1022. transformConstraint = Vector3(1.0, 0.0, 0.0);
  1023. setTransformPlane(true, false, false);
  1024. resetTransform();
  1025. } else {
  1026. switch(gizmoMode) {
  1027. case GIZMO_MODE_3D:
  1028. toggleOrientation();
  1029. break;
  1030. case GIZMO_MODE_2D_Y:
  1031. transformConstraint = Vector3(1.0, 0.0, 1.0);
  1032. break;
  1033. case GIZMO_MODE_2D_Z:
  1034. transformConstraint = Vector3(1.0, 1.0, 0.0);
  1035. break;
  1036. }
  1037. resetTransform();
  1038. }
  1039. }
  1040. }
  1041. }
  1042. break;
  1043. case KEY_y:
  1044. {
  1045. if(transforming && (mode == TRANSFORM_MOVE_VIEW || mode == TRANSFORM_SCALE_VIEW || ((mode == TRANSFORM_ROTATE_VIEW) && (gizmoMode == GIZMO_MODE_3D)))) {
  1046. if((coreInput->getKeyState(KEY_LSHIFT) || coreInput->getKeyState(KEY_RSHIFT)) && (mode != TRANSFORM_ROTATE_VIEW) && ((gizmoMode == GIZMO_MODE_2D_Y) || (gizmoMode == GIZMO_MODE_3D))) {
  1047. if(transformConstraint != Vector3(1.0, 0.0, 1.0)) {
  1048. transformConstraint = Vector3(1.0, 0.0, 1.0);
  1049. setTransformPlane(true, false, true);
  1050. resetTransform();
  1051. } else {
  1052. toggleOrientation();
  1053. resetTransform();
  1054. }
  1055. } else if(gizmoMode != GIZMO_MODE_2D_Y){
  1056. if(transformConstraint != Vector3(0.0, 1.0, 0.0)) {
  1057. transformConstraint = Vector3(0.0, 1.0, 0.0);
  1058. setTransformPlane(false, true, false);
  1059. resetTransform();
  1060. } else {
  1061. switch(gizmoMode) {
  1062. case GIZMO_MODE_3D:
  1063. toggleOrientation();
  1064. break;
  1065. case GIZMO_MODE_2D_X:
  1066. transformConstraint = Vector3(0.0, 1.0, 1.0);
  1067. break;
  1068. case GIZMO_MODE_2D_Z:
  1069. transformConstraint = Vector3(1.0, 1.0, 0.0);
  1070. break;
  1071. }
  1072. resetTransform();
  1073. }
  1074. }
  1075. }
  1076. }
  1077. break;
  1078. case KEY_z:
  1079. {
  1080. if(transforming && (mode == TRANSFORM_MOVE_VIEW || mode == TRANSFORM_SCALE_VIEW || ((mode == TRANSFORM_ROTATE_VIEW) && (gizmoMode == GIZMO_MODE_3D)))) {
  1081. if((coreInput->getKeyState(KEY_LSHIFT) || coreInput->getKeyState(KEY_RSHIFT)) && (mode != TRANSFORM_ROTATE_VIEW) && ((gizmoMode == GIZMO_MODE_2D_Z) || (gizmoMode == GIZMO_MODE_3D))) {
  1082. if(transformConstraint != Vector3(1.0, 1.0, 0.0)) {
  1083. transformConstraint = Vector3(1.0, 1.0, 0.0);
  1084. setTransformPlane(true, true, false);
  1085. resetTransform();
  1086. } else {
  1087. toggleOrientation();
  1088. resetTransform();
  1089. }
  1090. } else if(gizmoMode != GIZMO_MODE_2D_Z){
  1091. if(transformConstraint != Vector3(0.0, 0.0, 1.0)) {
  1092. transformConstraint = Vector3(0.0, 0.0, 1.0);
  1093. setTransformPlane(false, false, true);
  1094. resetTransform();
  1095. } else {
  1096. switch(gizmoMode) {
  1097. case GIZMO_MODE_3D:
  1098. toggleOrientation();
  1099. break;
  1100. case GIZMO_MODE_2D_X:
  1101. transformConstraint = Vector3(0.0, 1.0, 1.0);
  1102. break;
  1103. case GIZMO_MODE_2D_Y:
  1104. transformConstraint = Vector3(1.0, 0.0, 1.0);
  1105. break;
  1106. }
  1107. resetTransform();
  1108. }
  1109. }
  1110. }
  1111. }
  1112. break;
  1113. case KEY_ESCAPE:
  1114. {
  1115. if (mode == TRANSFORM_SCALE_VIEW || mode == TRANSFORM_ROTATE_VIEW || mode == TRANSFORM_MOVE_VIEW) {
  1116. dispatchEndEvent();
  1117. mode = previousMode;
  1118. }
  1119. resetTransform();
  1120. transforming = false;
  1121. firstMove = true;
  1122. }
  1123. break;
  1124. }
  1125. }
  1126. }
  1127. if(transforming) {
  1128. switch(event->getEventCode()) {
  1129. case InputEvent::EVENT_MOUSEMOVE:
  1130. {
  1131. switch(mode) {
  1132. case TRANSFORM_MOVE:
  1133. {
  1134. Vector3 newPoint;
  1135. if(isConstrainedToSingleAxis())
  1136. newPoint = getPositionAlongAxis();
  1137. else
  1138. newPoint = getTransformPlanePosition();
  1139. Vector3 diff = (planeMatrix.Inverse() * newPoint) -(planeMatrix.Inverse() * startingPoint);
  1140. diff = diff * getCompoundScale();
  1141. transformSelectedEntities(transformConstraint * diff, Vector3(0.0, 0.0, 0.0), 0.0);
  1142. startingPoint = newPoint;
  1143. }
  1144. break;
  1145. case TRANSFORM_SCALE_VIEW:
  1146. {
  1147. Vector2 pos2d = getScreenPosition(targetCamera->getProjectionMatrix(), targetCamera->getAnchorAdjustedMatrix(), targetCamera->getViewport());
  1148. Vector2 currentMouse = getCorrectedMousePosition();
  1149. Number scaleNew = currentMouse.distance(pos2d) / mouseStart2d.distance(pos2d);
  1150. Vector3 scaleVec = transformConstraint * scaleNew;
  1151. scaleAmount = scaleNew;
  1152. if((mouseStart2d - pos2d).dot(currentMouse - pos2d) < 0.0)
  1153. scaleVec = -scaleVec;
  1154. scaleVec.x = (transformConstraint.x == 0.0) ? 1.0 : scaleVec.x;
  1155. scaleVec.y = (transformConstraint.y == 0.0) ? 1.0 : scaleVec.y;
  1156. scaleVec.z = (transformConstraint.z == 0.0) ? 1.0 : scaleVec.z;
  1157. transformSelectedEntities(Vector3(0.0, 0.0, 0.0), scaleVec, 0.0);
  1158. }
  1159. break;
  1160. case TRANSFORM_ROTATE_VIEW:
  1161. {
  1162. Number newAngle = get2dAngle();
  1163. Vector3 newPoint = planeMatrix.Inverse() *getTransformPlanePosition();
  1164. transformSelectedEntities(Vector3(0.0, 0.0, 0.0), Vector3(0.0, 0.0, 0.0), newAngle - startingAngle);
  1165. startingAngle = newAngle;
  1166. }
  1167. break;
  1168. case TRANSFORM_MOVE_VIEW:
  1169. {
  1170. Vector3 newPoint;
  1171. if(isConstrainedToSingleAxis())
  1172. newPoint = getPositionAlongAxis();
  1173. else
  1174. newPoint = getTransformPlanePosition();
  1175. Vector3 diff = newPoint - startingPoint;
  1176. transformSelectedEntities(transformConstraint * diff, Vector3(0.0, 0.0, 0.0), 0.0);
  1177. startingPoint = newPoint;
  1178. }
  1179. break;
  1180. case TRANSFORM_SCALE:
  1181. {
  1182. Vector3 newPoint = getTransformPlanePosition();
  1183. Vector3 diff = (planeMatrix.Inverse() * newPoint) -(planeMatrix.Inverse() * startingPoint);
  1184. diff = diff * getCompoundScale();
  1185. transformSelectedEntities(Vector3(0.0, 0.0, 0.0), (diff * transformConstraint) / getCompoundScale().x, 0.0);
  1186. startingPoint = newPoint;
  1187. }
  1188. break;
  1189. case TRANSFORM_ROTATE:
  1190. {
  1191. Number newAngle = get2dAngle();
  1192. transformSelectedEntities(Vector3(0.0, 0.0, 0.0), Vector3(0.0, 0.0, 0.0), newAngle - startingAngle);
  1193. startingAngle = newAngle;
  1194. }
  1195. break;
  1196. }
  1197. }
  1198. break;
  1199. case InputEvent::EVENT_MOUSEUP:
  1200. {
  1201. dispatchEndEvent();
  1202. if(mode == TRANSFORM_SCALE_VIEW || mode == TRANSFORM_ROTATE_VIEW || mode == TRANSFORM_MOVE_VIEW) {
  1203. mode = previousMode;
  1204. }
  1205. if( orientation != startingOrientation && startingOrientation != -1) {
  1206. orientation = startingOrientation;
  1207. startingOrientation = -1;
  1208. }
  1209. transforming = false;
  1210. firstMove = true;
  1211. }
  1212. break;
  1213. }
  1214. }
  1215. }
  1216. }
  1217. void TransformGizmo::dispatchEndEvent() {
  1218. switch(mode) {
  1219. case TRANSFORM_MOVE:
  1220. case TRANSFORM_MOVE_VIEW:
  1221. dispatchEvent(new TrasnformGizmoEvent(mode), Event::CHANGE_EVENT);
  1222. break;
  1223. case TRANSFORM_SCALE:
  1224. case TRANSFORM_SCALE_VIEW:
  1225. dispatchEvent(new TrasnformGizmoEvent(mode), Event::CHANGE_EVENT);
  1226. break;
  1227. case TRANSFORM_ROTATE:
  1228. case TRANSFORM_ROTATE_VIEW:
  1229. dispatchEvent(new TrasnformGizmoEvent(mode), Event::CHANGE_EVENT);
  1230. break;
  1231. }
  1232. }
  1233. TransformGizmo::~TransformGizmo() {
  1234. coreInput->removeAllHandlersForListener(this);
  1235. }
  1236. void TransformGizmo::updateOrientationForEntity(Entity *entity) {
  1237. Quaternion q;
  1238. switch(orientation) {
  1239. case ORIENTATION_GLOBAL:
  1240. setRotationByQuaternion(q);
  1241. break;
  1242. case ORIENTATION_LOCAL:
  1243. if(gizmoMode == GIZMO_MODE_3D || mode == TRANSFORM_SCALE)
  1244. q = entity->getRotationQuat();
  1245. setRotationByQuaternion(q);
  1246. break;
  1247. }
  1248. }
  1249. void TransformGizmo::Update() {
  1250. if(selectedEntities.size() > 0) {
  1251. Vector3 centerPoint;
  1252. for(int i=0; i < selectedEntities.size(); i++) {
  1253. centerPoint += selectedEntities[i]->getConcatenatedMatrix().getPosition();
  1254. }
  1255. centerPoint = centerPoint / selectedEntities.size();
  1256. setPosition(centerPoint);
  1257. updateOrientationForEntity(selectedEntities[0]);
  1258. }
  1259. viewportRotateGripBase->setRotationByQuaternion(getRotationQuat().Inverse());
  1260. Number scale;
  1261. if(gizmoMode != GIZMO_MODE_3D) {
  1262. scale = targetCamera->getOrthoSizeX() * 0.1;
  1263. } else {
  1264. scale = getPosition().distance(targetCamera->getPosition()) * 0.1;
  1265. }
  1266. setScale(scale, scale, scale);
  1267. }