TransformGizmo.cpp 55 KB

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