*/}}

la_widgets_viewers.c 47 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046
  1. #include "../la_5.h"
  2. extern LA MAIN;
  3. extern struct _tnsMain *T;
  4. //============================================================= [Draw]
  5. extern tnsFontManager *FM;
  6. void la_RootObjectDrawFullscreenQuad(tnsOffscreen* DeferredOffScr,tnsCamera* c, real Aspect){
  7. real FOV = c->FOV;
  8. tnsMatrix44d mat;
  9. tnsGetMVMatrix(mat);
  10. real vv[3]={0,0,-1};
  11. real ViewDir[3]={0},vdTR[3]={0},vdBR[3]={0},vdTL[3]={0},vdBL[3]={0};
  12. tnsApplyRotation43d(ViewDir,mat,vv);
  13. real fovv=FOV/2, fovh=fovh=atan(tan(FOV/2)*Aspect);
  14. vv[0]=tan(fovh); vv[1]=tan(fovv); tnsApplyRotation43d(vdTR,mat,vv); // note don't normalize those.
  15. vv[0]=tan(fovh); vv[1]=tan(-fovv);tnsApplyRotation43d(vdBR,mat,vv);
  16. vv[0]=tan(-fovh);vv[1]=tan(-fovv);tnsApplyRotation43d(vdBL,mat,vv);
  17. vv[0]=tan(-fovh);vv[1]=tan(fovv); tnsApplyRotation43d(vdTL,mat,vv);
  18. tnsUseRayShader();
  19. tnsEnableShaderv(T->RayShader);
  20. tnsSetRayShaderUniformTextures(DeferredOffScr);
  21. real uv[12]; tnsMakeQuad3d(uv, LA_COLOR3(vdTL), LA_COLOR3(vdTR), LA_COLOR3(vdBR), LA_COLOR3(vdBL));
  22. real vt[8]; tnsMakeQuad2d(vt, -1,1, 1,1, 1,-1, -1,-1);
  23. tnsTexCoordArray3d(uv,4);
  24. tnsVertexArray2d(vt,4);
  25. tnsPackAs(GL_TRIANGLE_FAN);
  26. glUniform3f(T->RayShader->uViewDir,LA_COLOR3(ViewDir));
  27. glUniform3f(T->RayShader->uViewPos,LA_COLOR3(c->Base.Location));
  28. glUniform1f(T->RayShader->uFOV,FOV);
  29. tnsFlush();
  30. tnsUseImmShader();
  31. }
  32. #define LA_DEPTH_RESOLUTION 1024
  33. void la_RootObjectDraw(laBoxedTheme *bt, tnsObject *root, laUiItem* ui){
  34. laCanvasExtra *e = ui->Extra;
  35. tnsCamera *c = e->UsingCamera ? e->UsingCamera : e->ViewingCamera;
  36. int W, H;
  37. W = ui->R - ui->L;
  38. H = ui->B - ui->U;
  39. tnsFlush();
  40. if (!W || !H) return;
  41. if (!e->OffScr || e->OffScr->pColor[0]->Height != ui->B - ui->U || e->OffScr->pColor[0]->Width != ui->R - ui->L){
  42. if (e->OffScr) tnsDelete2DOffscreen(e->OffScr);
  43. if (e->DeferredOffScr) tnsDelete2DOffscreen(e->DeferredOffScr);
  44. e->OffScr = tnsCreate2DOffscreen(GL_RGBA, W, H, MAIN.PanelMultisample ,1);
  45. e->DeferredOffScr = tnsCreateDeferredOffscreen(W,H);
  46. }
  47. if (0){//(e->CurrentScene && e->CurrentScene->ActiveSun){
  48. //if(!e->OffScrShadow) e->OffScrShadow = tnsCreate2DOffscreen(0, LA_DEPTH_RESOLUTION, LA_DEPTH_RESOLUTION, MAIN.PanelMultisample, 1);
  49. //tnsUseNoTexture();
  50. //tnsDrawToOffscreen(e->OffScrShadow, 1, TNS_ATTACHMENT_ARRAY_NONE);
  51. //tnsViewportWithScissor(0,0,LA_DEPTH_RESOLUTION,LA_DEPTH_RESOLUTION);
  52. //glEnable(GL_DEPTH_TEST);
  53. //glClearDepth(1.0); glClear(GL_DEPTH_BUFFER_BIT);
  54. //tnsResetViewMatrix();tnsResetModelMatrix();tnsResetProjectionMatrix();
  55. //tnsEnableShaderv(T->ShadowShader);
  56. //tnsUseShadowShader();
  57. //tnsApplyShadowCameraView(e->CurrentScene->ActiveSun);
  58. //
  59. //real p[12];
  60. //tnsColor4dv(laThemeColor(bt,LA_BT_BORDER));
  61. //tnsMakeQuad3d(p, -100,-100, 25,-100,100, 25,100,100, 25,100,-100, 25);
  62. //tnsVertexArray3d(p, 4); tnsPackAs(GL_TRIANGLE_FAN);
  63. //tnsMakeQuad3d(p, -100,-100, 74,-100,100, 74,100,100, 74,100,-100, 74);
  64. //tnsVertexArray3d(p, 4); tnsPackAs(GL_TRIANGLE_FAN);
  65. ////tnsDrawFloor(e->GridSize, e->GridSpan, e->ShowAxis);
  66. //tnsColor4dv(laThemeColor(bt,LA_BT_BORDER));
  67. //tnsDrawFloor(e->GridSize, e->GridSpan, e->ShowAxis);
  68. //tnsEnableShaderv(T->immShader);
  69. //tnsFlush();
  70. tnsDrawToOffscreen(e->OffScr,1,0);
  71. tnsViewportWithScissor(0, 0, W, H);
  72. tnsClearColorv(laThemeColor(bt,LA_BT_NORMAL));
  73. tnsClearAll();
  74. //tnsUseSceneShader();
  75. //tnsEnableShaderv(T->SceneShader);
  76. //tnsApplyShadowCameraView(e->CurrentScene->ActiveSun);
  77. tnsApplyCameraView(W, H, c);
  78. //
  79. //tnsUseTexture(e->OffScrShadow->pDepth);
  80. //
  81. //tnsColor4dv(laThemeColor(bt,LA_BT_BORDER));
  82. //tnsMakeQuad3d(p, -100,-100, 25,-100,100, 25,100,100, 25,100,-100, 25);
  83. //tnsVertexArray3d(p, 4); tnsPackAs(GL_TRIANGLE_FAN);
  84. //tnsMakeQuad3d(p, -100,-100, 74,-100,100, 74,100,100, 74,100,-100, 74);
  85. //tnsVertexArray3d(p, 4); tnsPackAs(GL_TRIANGLE_FAN);
  86. //tnsUseNoTexture();
  87. //if (!e->LineDrawingMode && e->ShowFloorGrid){
  88. // tnsColor4dv(laThemeColor(bt,LA_BT_BORDER));
  89. // tnsDrawFloor(e->GridSize, e->GridSpan, e->ShowAxis);
  90. // tnsFlush();
  91. //}
  92. //
  93. //tnsFlush();
  94. }else if(e->OffScrShadow){ tnsDelete2DOffscreen(e->OffScrShadow); }
  95. //{
  96. // tnsDrawToOffscreen(e->OffScr, 1, 0);
  97. // //tnsUseShader(T->TEST_MatcapShader);
  98. // tnsEnableShaderv(T->TEST_MatcapShader);
  99. // tnsViewportWithScissor(0, 0, W, H);
  100. // tnsClearColorv(laThemeColor(bt,LA_BT_NORMAL));
  101. // tnsClearAll();
  102. //}
  103. tnsDrawToOffscreen(e->DeferredOffScr,3,TNS_ATTACHMENT_ARRAY_0_1_2);
  104. tnsViewportWithScissor(0, 0, W, H);
  105. tnsClearColorv(laThemeColor(bt,LA_BT_NORMAL)); tnsClearAll();
  106. tnsApplyCameraView(W, H, c);
  107. tnsPushMatrix();
  108. tnsPopMatrix(); //those are necessary when ui is the first in list;
  109. glEnable(GL_DEPTH_TEST); glDisable(GL_BLEND);
  110. tnsEnableShaderv(T->immShader);
  111. tnsUnbindTexture(); tnsUniformUseTexture(T->immShader,0,0); tnsUseMultiplyColor(0);
  112. glPointSize(6); glLineWidth(3);
  113. tnsDrawObjectTree(root, 0, 0, e->SelectMode);
  114. glPointSize(1); glLineWidth(2);
  115. glDepthMask(GL_FALSE); glPolygonMode(GL_FRONT_AND_BACK, GL_LINE); //glEnable(GL_POLYGON_OFFSET_LINE); glPolygonOffset(1,1);
  116. tnsDrawObjectTree(root, root->Active, 1, 0);
  117. glDepthMask(GL_TRUE); glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
  118. glLineWidth(1);
  119. //laInvoke(0,"M_select",e,&ui->ExtraPP,0,0);
  120. //glColorMask(GL_FALSE, GL_FALSE,GL_FALSE,GL_FALSE);
  121. //tnsDrawObjectTree(e->CurrentScene, 0);
  122. tnsFlush();
  123. //glColorMask(GL_TRUE, GL_TRUE, GL_TRUE, GL_TRUE);
  124. if (!e->LineDrawingMode && e->ShowFloorGrid){
  125. tnsUseNoTexture();
  126. real* color=laThemeColor(bt,LA_BT_BORDER); tnsColor4d(LA_COLOR3(color),0.4);
  127. tnsDrawFloor(e->GridSize, e->GridSpan, e->ShowAxis);
  128. tnsFlush();
  129. }
  130. glDisable(GL_DEPTH_TEST); glEnable(GL_BLEND);
  131. tnsDrawToOffscreen(e->OffScr, 1,0);
  132. tnsClearColorv(laThemeColor(bt,LA_BT_NORMAL)); tnsClearAll();
  133. la_RootObjectDrawFullscreenQuad(e->DeferredOffScr, c, (real)W/(real)H);
  134. }
  135. void la_RootObjectDrawOverlay(laUiItem *ui, int h){
  136. laCanvasExtra *e = ui->Extra;
  137. laBoxedTheme *bt = (*ui->Type->Theme);
  138. tnsDraw2DTextureDirectly(e->OffScr->pColor[0], ui->L, ui->U, ui->R - ui->L, ui->B - ui->U);
  139. //if(e->OffScrShadow){
  140. //tnsDraw2DTextureDirectly(e->OffScrShadow->pDepth, ui->L, ui->U, ui->R - ui->L, ui->B - ui->U);
  141. //}
  142. tnsUseNoTexture();
  143. if (e->DrawCursor){
  144. if(e->DrawCursor==LA_CANVAS_CURSOR_ARROW){
  145. tnsColor4dv(laThemeColor(bt,LA_BT_TEXT));
  146. tnsVertex2d(e->OnX, e->OnY); tnsVertex2d(e->TargetX, e->TargetY);
  147. tnsPackAs(GL_LINES);
  148. }else{
  149. if(e->DrawCursor==LA_CANVAS_CURSOR_BOX){
  150. tnsColor4dv(laAccentColor(LA_BT_NORMAL));
  151. tnsVertex2d(e->ClickedX, e->ClickedY); tnsVertex2d(e->ClickedX, e->OnY);
  152. tnsVertex2d(e->OnX, e->OnY); tnsVertex2d(e->OnX, e->ClickedY);
  153. tnsPackAs(GL_TRIANGLE_FAN);
  154. tnsColor4dv(laAccentColor(LA_BT_TEXT));
  155. tnsVertex2d(e->ClickedX, e->ClickedY); tnsVertex2d(e->ClickedX, e->OnY);
  156. tnsVertex2d(e->OnX, e->OnY); tnsVertex2d(e->OnX, e->ClickedY);
  157. tnsPackAs(GL_LINE_LOOP);
  158. }
  159. tnsColor4dv(laThemeColor(bt,LA_BT_TEXT));
  160. tnsVertex2d(e->OnX, ui->U); tnsVertex2d(e->OnX, ui->B);
  161. tnsVertex2d(ui->L, e->OnY); tnsVertex2d(ui->R, e->OnY);
  162. tnsPackAs(GL_LINES);
  163. }
  164. }
  165. if(MAIN.CurrentWindow->MaximizedUi!=ui){
  166. tnsColor4dv(laThemeColor(bt,LA_BT_BORDER));
  167. tnsVertex2d(ui->L, ui->U);
  168. tnsVertex2d(ui->R, ui->U);
  169. tnsVertex2d(ui->R, ui->B);
  170. tnsVertex2d(ui->L, ui->B);
  171. tnsPackAs(GL_LINE_LOOP);
  172. }
  173. tnsFlush();
  174. char* MaxIco=(MAIN.CurrentWindow->MaximizedUi!=ui)?"🡹":"🡻";
  175. tnsVector4d color; tnsVectorCopy4d(laThemeColor(bt,LA_BT_TEXT),color);
  176. color[3]=ui->State==LA_UI_NORMAL?0.0:(ui->State==LA_UI_ACTIVE?0.5:1.0);
  177. int startx=(ui->R+ui->L)/2-LA_RH2*2;
  178. tnsDrawStringAuto("☰",color,startx, startx+LA_RH, ui->B-bt->BM-LA_RH, LA_TEXT_ALIGN_CENTER);
  179. tnsDrawStringAuto(MaxIco,color,startx+LA_RH, startx+LA_2RH, ui->B-bt->BM-LA_RH, LA_TEXT_ALIGN_CENTER);
  180. if(ui->Expand>=0 && ui!=MAIN.CurrentWindow->MaximizedUi)
  181. tnsDrawStringAuto("◿",laThemeColor(bt,LA_BT_BORDER),ui->R-LA_RH, ui->R, ui->B-bt->BM-LA_RH, LA_TEXT_ALIGN_CENTER);
  182. }
  183. void la_CanvasDrawTexture(laBoxedTheme *bt, tnsTexture *t, laUiItem* ui){
  184. laCanvasExtra* e=ui->Extra;
  185. int W, H; W = ui->R - ui->L; H = ui->B - ui->U;
  186. if (W<=0 || H<=0) return;
  187. tnsFlush();
  188. if (!e->OffScr || e->OffScr->pColor[0]->Height != ui->B - ui->U || e->OffScr->pColor[0]->Width != ui->R - ui->L){
  189. if (e->OffScr) tnsDelete2DOffscreen(e->OffScr);
  190. e->OffScr = tnsCreate2DOffscreen(GL_RGBA, W, H, MAIN.PanelMultisample, 1);
  191. }
  192. tnsDrawToOffscreen(e->OffScr, 1, 0);
  193. tnsViewportWithScissor(0, 0, W, H);
  194. tnsResetViewMatrix();tnsResetModelMatrix();tnsResetProjectionMatrix();
  195. tnsOrtho(e->PanX - W * e->ZoomX / 2, e->PanX + W * e->ZoomX / 2, e->PanY - e->ZoomY * H / 2, e->PanY + e->ZoomY * H / 2, 100, -100);
  196. //glClearColor(0.3,0.3,0.3, 1);
  197. if (e->ClearBackground){ tnsClearColorv(laThemeColor(bt,LA_BT_NORMAL)); glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); }
  198. // above is basic routine setup
  199. real V[8] = {0};
  200. real UV[8] = {0};
  201. if (t){
  202. real w2=t->Width/2, h2=t->Height/2;
  203. tnsDraw2DTextureDirectly(t, -w2, h2, t->Width, -t->Height);
  204. tnsFlush();
  205. if (e->ImageDrawBorder){
  206. tnsUseNoTexture();
  207. tnsMakeQuad2d(V, -w2, -h2, -w2, h2, w2, h2, w2, -h2);
  208. tnsVertexArray2d(V, 4);
  209. tnsColor4dv(laThemeColor(bt,LA_BT_BORDER));
  210. tnsVertexArray2d(V, 4);
  211. tnsPackAs(GL_LINE_LOOP);
  212. tnsFlush();
  213. }
  214. }
  215. }
  216. real la_CanvasDrawGetDisplayStep(real Max, real Min, int AreaSize){
  217. real r;
  218. real Step = 1.0;
  219. real DispW;
  220. real Div[3] = {2.0, 2.5, 2.0};
  221. int Times = 302;
  222. while (1){
  223. Times++;
  224. r = tnsGetRatiod(0, Max - Min, Step);
  225. DispW = tnsLinearItp(0, AreaSize, r);
  226. if (DispW > 100){
  227. Step /= Div[Times % 3];
  228. }elif (DispW < 40){
  229. Step *= Div[Times % 3];
  230. }else break;
  231. }
  232. return Step;
  233. }
  234. void la_CanvasDrawGridW(real Ratio, int RealW, real Min, real Max, real U, real B){
  235. real t;
  236. real Step = la_CanvasDrawGetDisplayStep(Max, Min, RealW);
  237. t = (real)((int)(Min / Step)) * Step;
  238. for (t; t <= Max; t += Step){
  239. tnsVertex2d(t, U);
  240. tnsVertex2d(t, B);
  241. tnsPackAs(GL_LINES);
  242. }
  243. }
  244. void la_CanvasDrawGridH(real Ratio, int RealH, real Min, real Max, real L, real R){
  245. real t;
  246. real Step = la_CanvasDrawGetDisplayStep(Max, Min, RealH);
  247. t = (real)((int)(Min / Step)) * Step;
  248. for (t; t <= Max; t += Step){
  249. tnsVertex2d(L, t);
  250. tnsVertex2d(R, t);
  251. tnsPackAs(GL_LINES);
  252. }
  253. }
  254. void la_CanvasDrawRulerW(real Ratio, laBoxedTheme *bt, int RealW, real Min, real Max, int B){
  255. char buf[20] = {0};
  256. real t, IL;
  257. real Step = la_CanvasDrawGetDisplayStep(Max, Min, RealW);
  258. int U = B - LA_RH;
  259. tnsUseNoTexture();
  260. tnsColor4dv(laThemeColor(bt,LA_BT_BORDER));
  261. tnsVertex2d(0, U);
  262. tnsVertex2d(RealW, U);
  263. tnsVertex2d(RealW, B);
  264. tnsVertex2d(0, B);
  265. tnsPackAs(GL_TRIANGLE_FAN);
  266. tnsColor4dv(laThemeColor(bt,LA_BT_TEXT_ACTIVE));
  267. tnsVertex2d(0, U + 0.5);
  268. tnsVertex2d(RealW, U + 0.5);
  269. tnsPackAs(GL_LINES);
  270. t = (real)((int)(Min / Step)) * Step;
  271. for (t; t <= Max; t += Step){
  272. IL = tnsGetRatiod(Min, Max, t) * (RealW);
  273. sprintf(buf, "%.0f", t);
  274. tnsDrawStringAuto(buf, bt->Active, IL, IL + 100, U, 0);
  275. }
  276. }
  277. void la_CanvasDrawRulerRWithBkg(real Ratio, laBoxedTheme *bt, int RealH, real Min, real Max, int R){
  278. real t, OrigT;
  279. char buf[50][20] = {0};
  280. real IL[50];
  281. int i = 0;
  282. int W = 0, MaxW = 0;
  283. int RH = LA_RH, hH = RH / 2;
  284. int L;
  285. real Step = la_CanvasDrawGetDisplayStep(Max, Min, RealH);
  286. int IS = Step > 1 ? 0 : (Step > 0.09 ? 1 : 2);
  287. t = OrigT = (real)((int)(Min / Step)) * Step;
  288. for (t; t <= Max; t += Step, i++){
  289. sprintf(buf[i], "%.*f", IS, t);
  290. W = tnsStringGetWidth(buf[i], 0, 0);
  291. if (W > MaxW) MaxW = W;
  292. }
  293. L = R - MaxW - 4;
  294. tnsUseNoTexture();
  295. for (t = OrigT, i = 0; t <= Max; t += Step, i++){
  296. IL[i] = tnsGetRatiod(Max, Min, t) * (RealH);
  297. tnsColor4dv(laThemeColor(bt,LA_BT_BORDER));
  298. tnsVertex2d(R, IL[i] - hH);
  299. tnsVertex2d(R, IL[i] + hH);
  300. tnsVertex2d(L, IL[i] + hH);
  301. tnsVertex2d(L, IL[i] - hH);
  302. tnsPackAs(GL_TRIANGLE_FAN);
  303. tnsColor4dv(laThemeColor(bt,LA_BT_TEXT_ACTIVE));
  304. tnsVertex2d(R, IL[i] - hH);
  305. tnsVertex2d(R, IL[i] + hH);
  306. tnsVertex2d(L, IL[i] + hH);
  307. tnsVertex2d(L, IL[i] - hH);
  308. tnsPackAs(GL_LINE_LOOP);
  309. }
  310. for (t = OrigT, i = 0; t <= Max; t += Step, i++){
  311. tnsDrawStringAuto(buf[i], bt->Active, L + 2, R, IL[i] - hH, 0);
  312. }
  313. }
  314. void la_CanvasDraw(laUiItem *ui, int h){
  315. laBoxedTheme *bt = (*ui->Type->Theme);
  316. if (ui->CanvasTemplate && ui->CanvasTemplate->Draw) ui->CanvasTemplate->Draw(bt, ui->PP.EndInstance, ui);
  317. }
  318. void la_CanvasDrawOverlay(laUiItem *ui, int h){
  319. laCanvasExtra *e = ui->Extra;
  320. laBoxedTheme *bt = (*ui->Type->Theme);
  321. tnsDraw2DTextureDirectly(e->OffScr->pColor[0], ui->L, ui->U, ui->R - ui->L, ui->B - ui->U);
  322. tnsUseNoTexture();
  323. if(MAIN.CurrentWindow->MaximizedUi!=ui){
  324. tnsColor4dv(laThemeColor(bt,LA_BT_BORDER));
  325. tnsVertex2d(ui->L, ui->U);
  326. tnsVertex2d(ui->R, ui->U);
  327. tnsVertex2d(ui->R, ui->B);
  328. tnsVertex2d(ui->L, ui->B);
  329. tnsPackAs(GL_LINE_LOOP);
  330. }
  331. if (e->DrawCursor){
  332. tnsColor4dv(laThemeColor(bt,LA_BT_TEXT));
  333. tnsVertex2d(e->OnX, ui->U);
  334. tnsVertex2d(e->OnX, ui->B);
  335. tnsVertex2d(ui->L, e->OnY);
  336. tnsVertex2d(ui->R, e->OnY);
  337. tnsPackAs(GL_LINES);
  338. tnsFlush();
  339. }
  340. char* MaxIco=(MAIN.CurrentWindow->MaximizedUi!=ui)?"🡹":"🡻";
  341. tnsVector4d color; tnsVectorCopy4d(laThemeColor(bt,LA_BT_TEXT),color);
  342. color[3]=ui->State==LA_UI_NORMAL?0.0:(ui->State==LA_UI_ACTIVE?0.5:1.0);
  343. int startx=(ui->R+ui->L)/2-LA_RH2*2;
  344. tnsDrawStringAuto("☰",color,startx, startx+LA_RH, ui->B-bt->BM-LA_RH, LA_TEXT_ALIGN_CENTER);
  345. tnsDrawStringAuto(MaxIco,color,startx+LA_RH, startx+LA_2RH, ui->B-bt->BM-LA_RH, LA_TEXT_ALIGN_CENTER);
  346. if(ui->Expand>=0 && ui!=MAIN.CurrentWindow->MaximizedUi)
  347. tnsDrawStringAuto("◿",laThemeColor(bt,LA_BT_BORDER),ui->R-LA_RH, ui->R, ui->B-bt->BM-LA_RH, LA_TEXT_ALIGN_CENTER);
  348. tnsFlush();
  349. }
  350. int la_AnimateUiListRecursive(laUiList *uil);
  351. void laDefault3DViewOverlay(laUiItem *ui){
  352. laUiList *uil, *gu;
  353. laColumn *c, *cl, *cll, *clr, *cr, *crl, *crr, *gc, *gcl, *gcr;
  354. laPropPack *e = &ui->ExtraPP;
  355. laUiItem *b, *g;
  356. if (!(uil = ui->Subs.pFirst)) uil = laAddTabPage(ui, "New Group");
  357. c = laFirstColumn(uil);
  358. laSplitColumn(uil, c, 0.8);
  359. cl = laLeftColumn(c, 0);
  360. cr = laRightColumn(c, 0);
  361. laSplitColumn(uil, cl, 0.25);
  362. cll = laLeftColumn(cl, 0);
  363. clr = laRightColumn(cl, 0);
  364. laShowColumnAdjuster(uil, c);
  365. b=laBeginRow(uil,cl,0,0);
  366. laShowItem(uil,cl,&ui->ExtraPP,"select_mode")->Flags|=LA_UI_FLAGS_EXPAND;
  367. laShowItem(uil,cl,&ui->ExtraPP,"select_through");
  368. laEndRow(uil,b);
  369. g = laMakeFoldableGroup(uil, cll, "Tools", 0, 1);{
  370. gu = g->Page; gc = laFirstColumn(gu);
  371. laSplitColumn(gu, gc, 0.35); gcl = laLeftColumn(gc, 0); gcr = laRightColumn(gc, 0);
  372. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_delete");
  373. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_make_parent");
  374. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_unparent");
  375. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_add");
  376. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_grab");
  377. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_scale");
  378. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_rotate");
  379. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_clear_transformations");
  380. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_extrude");
  381. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_make");
  382. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_subdiv");
  383. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_add");
  384. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_separate");
  385. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_combine");
  386. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_duplicate");
  387. laShowItem(gu,gc,&ui->ExtraPP,"_this_M_recalculate_normals");
  388. }
  389. g = laMakeFoldableGroup(uil, cr, "Scene Info", 0, 1);{
  390. gu = g->Page;
  391. gc = laFirstColumn(gu);
  392. laSplitColumn(gu, gc, 0.35);
  393. gcl = laLeftColumn(gc, 0);
  394. gcr = laRightColumn(gc, 0);
  395. laShowItemFull(gu, gc, &ui->PP, 0, LA_WIDGET_COLLECTION_SELECTOR,0,laui_IdentifierOnly,0);
  396. laShowLabel(gu, gcl, "Rename:", 0, 0);
  397. laShowItem(gu, gcr, &ui->PP, "name");
  398. }
  399. g = laMakeFoldableGroup(uil, cr, "Object Info", 0, 0);{
  400. gu = g->Page;
  401. gc = laFirstColumn(gu);
  402. laSplitColumn(gu, gc, 0.35);
  403. gcl = laLeftColumn(gc, 0);
  404. gcr = laRightColumn(gc, 0);
  405. laShowItem(gu, gc, &ui->PP, "active.name");
  406. laShowLabel(gu, gcl, "Location:", 0, 0);laShowItem(gu, gcr, &ui->PP, "active.location")->Flags|=LA_UI_FLAGS_TRANSPOSE;
  407. laShowSeparator(gu,gc);
  408. laShowLabel(gu, gcl, "Rotation:", 0, 0);laShowItem(gu, gcr, &ui->PP, "active.rotation")->Flags|=LA_UI_FLAGS_TRANSPOSE;
  409. laShowItem(gu, gc, &ui->PP, "active.scale");
  410. }
  411. g = laMakeFoldableGroup(uil, cr, "Display", 0, 1);{
  412. gu = g->Page;
  413. gc = laFirstColumn(gu);
  414. laSplitColumn(gu, gc, 0.35);
  415. gcl = laLeftColumn(gc, 0);
  416. gcr = laRightColumn(gc, 0);
  417. laShowLabel(gu, gcl, "Axis:", 0, 0);
  418. laWidget w={_LA_UI_ENUM_SELECTOR, LA_UI_FLAGS_ICON|LA_UI_FLAGS_CYCLE|LA_UI_FLAGS_TRANSPOSE};
  419. laShowItemFull(gu, gcr, e, "show_axis",&w, "show_prefix=true;" ,0,0);
  420. laShowLabel(gu, gcl, "Floor Grid:", 0, 0);
  421. laShowItemFull(gu, gcr, e, "show_floor_grid", LA_WIDGET_ENUM_CYCLE, 0, 0, 0);
  422. laShowItem(gu, gc, e, "grid_size");
  423. laShowItem(gu, gc, e, "grid_span");
  424. //laShowLabel(gu, gcl, "NPR:", 0, 0);
  425. //laShowItemFull(gu, gcr, e, "npr_line_mode", LA_WIDGET_ENUM_CYCLE, 0, 0, 0);
  426. }
  427. }
  428. void la_3DViewInit(laUiItem *ui){
  429. laCanvasExtra *e = ui->Extra;
  430. if (e){
  431. return;
  432. }else{
  433. e = memAcquireHyper(sizeof(laCanvasExtra));
  434. }
  435. ui->Extra = e;
  436. e->ParentUi = ui;
  437. tnsVector3d pos={-15, -20, 7};
  438. tnsCamera* c= tnsCreateCamera(0, "VIEWING_CAMERA", rad(50), LA_COLOR3(pos), rad(70), 0, rad(-40), 25);
  439. memAssignRef(e,&e->ViewingCamera,c);
  440. tnsVector3d target={0,0,0}; tnsVector3d up={0,0,1};
  441. tnsLookAt(e->ViewingCamera, target, up);
  442. e->ViewingCamera->FocusDistance=tnsDist3dv(pos,target);
  443. e->GridSize = 10;
  444. e->GridSpan = 15;
  445. e->ShowAxis[0] = 1;
  446. e->ShowAxis[1] = 1;
  447. e->ShowFloorGrid = 1;
  448. //e->DisplayMode = LA_3D_VIEW_DISPLAY_WIRE;
  449. e->HeightCoeff = 10;
  450. laFirstColumn(laAddTabPage(ui, "New Group"));
  451. }
  452. void la_3DViewDestroy(laUiItem *ui){
  453. laCanvasExtra *e = ui->Extra;
  454. tnsDelete2DOffscreen(e->OffScr);
  455. tnsDelete2DOffscreen(e->OffScrShadow);
  456. //XXX: cause breaks tnsDestroyObject(e->ViewingCamera);
  457. memFree(e);
  458. }
  459. void laDefault2DViewOverlayRight(laUiItem *ui){
  460. laUiList *uil, *gu;
  461. laColumn *c, *cr, *crl, *crr, *gc, *gcl, *gcr, *cl;
  462. laPropPack *e = &ui->ExtraPP;
  463. laUiItem *b, *b1, *b2, *g;
  464. if (!(uil = ui->Subs.pFirst)) uil = laAddTabPage(ui, "New Group");
  465. uil->HeightCoeff=-1;
  466. c = laFirstColumn(uil);
  467. laSplitColumn(uil, c, 0.7);
  468. cl = laLeftColumn(c, 0);
  469. cr = laRightColumn(c, 0);
  470. laShowColumnAdjuster(uil, c);
  471. b=laBeginRow(uil,cl,0,0);
  472. laShowLabel(uil, cl, "Line Width:", 0, 0);
  473. laShowItem(uil, cl, e, "adaptive_line_width")->Flags|=LA_UI_FLAGS_PLAIN;
  474. laEndRow(uil,b);
  475. b=laBeginRow(uil,cl,0,0);
  476. laShowItem(uil, cl, &ui->PP, "size")->Flags|=LA_UI_FLAGS_PLAIN|LA_UI_FLAGS_TRANSPOSE|LA_TEXT_ALIGN_LEFT;
  477. laEndRow(uil,b);
  478. laShowItem(uil, cl, &ui->PP, "internal_type")->Flags|=LA_UI_FLAGS_PLAIN;
  479. g = laMakeFoldableGroup(uil, cr, "Display", 0, 1);{
  480. gu = g->Page;
  481. gc = laFirstColumn(gu);
  482. laSplitColumn(gu, gc, 0.35);
  483. gcl = laLeftColumn(gc, 0);
  484. gcr = laRightColumn(gc, 0);
  485. // needs another mode for access the ExtraPP.
  486. // laShowItem(gu, gc, &ui->ExtraPP, "maximize");
  487. laShowLabel(gu, gcl, "Transparency:", 0, 0);
  488. laShowItem(gu, gcr, e, "draw_image_alpha");
  489. laShowLabel(gu, gcl, "Border:", 0, 0);
  490. laShowItemFull(gu, gcr, e, "draw_image_border", LA_WIDGET_ENUM_CYCLE, 0, 0, 0);
  491. laShowLabel(gu, gcl, "Line Width:", 0, 0);
  492. laShowItemFull(gu, gcr, e, "adaptive_line_width", LA_WIDGET_ENUM_CYCLE, 0, 0, 0);
  493. laShowLabel(gu, gcl, "Cursor:", 0, 0);
  494. laShowItemFull(gu, gcr, e, "draw_cursor", LA_WIDGET_ENUM_CYCLE, 0, 0, 0);
  495. }laEndFoldableGroup(uil,g);
  496. }
  497. void la_CanvasUiInit(laUiItem* ui){ laCanvasTemplate* ct=ui->CanvasTemplate; if(ct->Init) ct->Init(ui); }
  498. void la_CanvasUiDestroy(laUiItem *ui){ laCanvasTemplate* ct=ui->CanvasTemplate; if(ct->Destroy) ct->Destroy(ui); }
  499. void la_CanvasInit(laUiItem *ui){
  500. laCanvasExtra *e = ui->Extra;
  501. if (!e){ e = memAcquireHyper(sizeof(laCanvasExtra)); ui->Extra = e; }
  502. e->ParentUi = ui;
  503. e->HeightCoeff = 10;
  504. e->ZoomX = 1;
  505. e->ZoomY = 1;
  506. e->ImageDrawAlpha = 1;
  507. e->ImageDrawBorder = 1;
  508. e->AdaptiveLineWidth = 1;
  509. e->ClearBackground = 1;
  510. laFirstColumn(laAddTabPage(ui, "New Group"));
  511. }
  512. void la_CanvasDestroy(laUiItem *ui){
  513. laCanvasExtra *e = ui->Extra;
  514. tnsDelete2DOffscreen(e->OffScr);
  515. memFree(e);
  516. }
  517. //============================================================= [Operators]
  518. int OPINV_3DOr2DViewUiItem(laOperator *a, laEvent *e){
  519. laUiItem *ui = a->Instance;
  520. a->CustomData = ui->Extra;
  521. return LA_RUNNING;
  522. }
  523. int OPEXT_3DOr2DViewUiItem(laOperator *a, int ExitCode){
  524. return 0;
  525. }
  526. int la_CanvasDetectButtons(laUiItem* ui, laBoxedTheme* bt, int x,int y, int total_buttons, int* very_close){
  527. int midx=(ui->R+ui->L)/2-LA_RH2*(total_buttons-1), midy=ui->B-LA_RH2-bt->BM;
  528. if(y>ui->B-LA_2RH-bt->BM || y<ui->U+LA_2RH+bt->TM || x<ui->L+LA_2RH+bt->TM || x>ui->R-LA_2RH-bt->RM)
  529. { if(very_close)*very_close=1; }else return 0;
  530. if(y<ui->B-LA_RH-bt->BM)return 0;
  531. for(int i=0;i<total_buttons;i++){ if(x>=midx-LA_RH2 && x<midx+LA_RH2) return i+1; midx+=LA_RH;}
  532. return 0;
  533. }
  534. int OPMOD_Canvas(laOperator *a, laEvent *e){
  535. laUiItem *ui = a->Instance;
  536. laBoxedTheme *bt = (*ui->Type->Theme);
  537. laCanvasExtra *ex = a->CustomData;
  538. laUiList *subu;
  539. laUiItem *subui = 0;
  540. laListHandle Locals = {0};
  541. int px = e->x, py = e->y;
  542. if (!laIsInUiItem(ui, e->x, e->y) && !ex->Dragging){
  543. ex->OnX = INT_MAX; ex->OnY = INT_MAX; ui->State=LA_UI_NORMAL;
  544. return LA_FINISHED_PASS;
  545. }
  546. //int x=e->x,y=e->y; laPanel*p=MAIN.CurrentPanel,*dp;
  547. //laLocalToWindow(0, p, &x, &y); if(!p) p=MAIN.CurrentWindow->MaximizedUiPanel;
  548. //if(dp = laDetectPanel(x,y) && dp!=p){ return LA_RUNNING_PASS; }
  549. if(e->Type&LA_MOUSE_EVENT){
  550. if(e->Type == LA_L_MOUSE_DOWN && ui->Expand>=0 && e->x>ui->R-bt->RM-LA_RH && e->y>ui->B-bt->BM-LA_RH){
  551. ex->Dragging=1; ex->ClickedX=e->x; ex->ClickedY=e->y; ex->TargetIndexVali=ui->Expand;
  552. return LA_RUNNING;
  553. }
  554. if(ex->Dragging){
  555. if(e->Type==LA_MOUSEMOVE){ ui->Expand=ex->TargetIndexVali+((real)e->y-ex->ClickedY+0.5)/LA_RH;
  556. if(ex->HeightCoeff<1) ex->HeightCoeff=1; printf("%d\n",ex->HeightCoeff);laRecalcCurrentPanel(); }
  557. elif(e->Type==LA_L_MOUSE_UP){ ex->Dragging=0; }
  558. elif(e->Type==LA_R_MOUSE_DOWN){ ex->Dragging=0; ui->Expand=ex->TargetIndexVali; laRecalcCurrentPanel(); }
  559. return LA_RUNNING;
  560. }
  561. int VeryClose=0; int btn=la_CanvasDetectButtons(ui, bt, e->x, e->y, 2, &VeryClose);
  562. if(e->Type==LA_L_MOUSE_DOWN){
  563. if(btn==2){ if(MAIN.CurrentWindow->MaximizedUi==ui) laRestoreCanvasUI(); else laMaximizeCanvasUI(ui,MAIN.CurrentPanel); return LA_RUNNING; }
  564. if(btn==1){
  565. if(ui->Flags&LA_UI_FLAGS_NO_OVERLAY)ui->Flags&=~LA_UI_FLAGS_NO_OVERLAY; else ui->Flags|=LA_UI_FLAGS_NO_OVERLAY;
  566. laRedrawCurrentPanel(); return LA_RUNNING;
  567. }
  568. }
  569. int state=VeryClose?(btn?LA_UI_EDITING:LA_UI_ACTIVE):LA_UI_NORMAL;
  570. if(state!=ui->State){ ui->State=state;laRedrawCurrentPanel(); }
  571. }
  572. //laLocalToPanel(a, &px, &py);
  573. if(!(ui->Flags&LA_UI_FLAGS_NO_OVERLAY)){
  574. for (subu = ui->Subs.pFirst; subu; subu = subu->Item.pNext){
  575. if (subui = la_DetectUiItemRecursive(subu, px, py, ui->B, &Locals, 0)){
  576. if (subui && !a->Child && subui->Type->OperatorType){
  577. //printf("invui %d\n", tui);
  578. laUiList *luil = ((laUiListDrawItem *)Locals.pFirst)->Target;
  579. laSetOperatorLocalizer(a->ToPanel);
  580. if (laInvokeUiP(a, subui->Type->OperatorType, e, subui, &Locals, 0) >= 0) laRetriggerOperators();
  581. lstClearPointer(&Locals);
  582. return LA_RUNNING_PASS;
  583. }
  584. }
  585. lstClearPointer(&Locals);
  586. }
  587. }
  588. if (ex->DrawCursor){
  589. ex->OnX = e->x;
  590. ex->OnY = e->y;
  591. laRedrawCurrentPanel();
  592. }
  593. if (laKeyMapExecuteEventEx(a, &ui->ExtraPP, &ui->CanvasTemplate->KeyMapper, e)) return LA_RUNNING;
  594. if (laKeyMapExecuteEventEx(a, &ui->ExtraPP, &ui->Type->KeyMapper, e)) return LA_RUNNING;
  595. return LA_RUNNING_PASS;
  596. }
  597. int OPCHK_Is3DViewExtra(laPropPack *This, laStringSplitor *ss){
  598. if (This && This->LastPs->p->SubProp == _LA_PROP_3D_EXTRA) return 1;
  599. return 0;
  600. }
  601. int OPINV_3DViewCameraZoom(laOperator *a, laEvent *e){
  602. laCanvasExtra *ex = a->This->EndInstance;
  603. tnsCamera *c = ex->ViewingCamera;
  604. if (strArgumentMatch(a->ExtraInstructionsP, "direction", "in")){
  605. tnsZoomViewingCamera(c, 0.1);
  606. laRedrawCurrentPanel();
  607. }elif (strArgumentMatch(a->ExtraInstructionsP, "direction", "out")){
  608. tnsZoomViewingCamera(c, -0.1);
  609. laRedrawCurrentPanel();
  610. }
  611. return LA_FINISHED;
  612. }
  613. int OPINV_3DOr2DViewAdjust(laOperator *a, laEvent *e){
  614. laGeneralUiExtraData *ex = memAcquireSimple(sizeof(laGeneralUiExtraData));
  615. laSetWindowCursor(LA_HAND);
  616. a->CustomData = ex;
  617. ex->LastX = e->x;
  618. ex->LastY = e->y;
  619. printf("%d %d\n",e->x,e->y);
  620. return LA_RUNNING;
  621. }
  622. int OPEXT_3DOr2DViewAdjust(laOperator *a, int Mark){
  623. if (a->CustomData) memFree(a->CustomData);
  624. laSetWindowCursor(LA_ARROW);
  625. }
  626. int OPMOD_3DViewCameraRotate(laOperator *a, laEvent *e){
  627. laCanvasExtra *ex = a->This->EndInstance;
  628. laGeneralUiExtraData *uex = a->CustomData;
  629. if (e->Type == LA_L_MOUSE_UP || e->Type == LA_M_MOUSE_UP || e->Type == LA_R_MOUSE_DOWN) return LA_FINISHED;
  630. if (e->Type == LA_MOUSEMOVE){
  631. tnsRotateViewingCamera(ex->ViewingCamera, (real)(uex->LastY - e->y) / 150.0, (real)(uex->LastX - e->x) / 150.0);
  632. uex->LastX = e->x;
  633. uex->LastY = e->y;
  634. laRedrawCurrentPanel();
  635. }
  636. return LA_RUNNING;
  637. }
  638. int OPMOD_3DViewCameraMove(laOperator *a, laEvent *e){
  639. laCanvasExtra *ex = a->This->EndInstance;
  640. laGeneralUiExtraData *uex = a->CustomData;
  641. if (e->Type == LA_L_MOUSE_UP || e->Type == LA_M_MOUSE_UP || e->Type == LA_R_MOUSE_DOWN){
  642. laSetWindowCursor(LA_ARROW);
  643. return LA_FINISHED;
  644. }
  645. if (e->Type == LA_MOUSEMOVE){
  646. tnsTranslateViewingCamera(ex->ViewingCamera, ex->ParentUi->R - ex->ParentUi->L, ex->ParentUi->B - ex->ParentUi->U, -e->x + uex->LastX, e->y - uex->LastY);
  647. //tnsTranslateObjectLocal(ex->ViewingCamera,
  648. // -(real)(e->x - uex->LastX) * ex->ViewingCamera->FocusDistance / 100,
  649. // (real)(e->y - uex->LastY) * ex->ViewingCamera->FocusDistance / 100,
  650. // 0);
  651. uex->LastX = e->x;
  652. uex->LastY = e->y;
  653. laRedrawCurrentPanel();
  654. }
  655. return LA_RUNNING;
  656. }
  657. int OPCHK_Is2DViewExtra(laPropPack *This, laStringSplitor *ss){
  658. if (This && (This->LastPs->p->SubProp == _LA_PROP_2D_EXTRA)) return 1;
  659. return 0;
  660. }
  661. int OPINV_CanvasZoom(laOperator *a, laEvent *e){
  662. laCanvasExtra *ex = a->This->EndInstance;
  663. if (strArgumentMatch(a->ExtraInstructionsP, "mode", "mouse")){
  664. laGeneralUiExtraData *ex = memAcquireSimple(sizeof(laGeneralUiExtraData));
  665. laSetWindowCursor(LA_HAND);
  666. a->CustomData = ex;
  667. ex->LastX = e->x;
  668. ex->LastY = e->y;
  669. return LA_RUNNING;
  670. }
  671. if (strArgumentMatch(a->ExtraInstructionsP, "direction", "in")){
  672. if (strArgumentMatch(a->ExtraInstructionsP, "axis", "x")){
  673. ex->ZoomX *= 0.9;
  674. }else{
  675. ex->ZoomX *= 0.9;
  676. ex->ZoomY *= 0.9;
  677. }
  678. laRedrawCurrentPanel();
  679. }elif (strArgumentMatch(a->ExtraInstructionsP, "direction", "out")){
  680. if (strArgumentMatch(a->ExtraInstructionsP, "axis", "x")){
  681. ex->ZoomX *= 1.1;
  682. }else{
  683. ex->ZoomX *= 1.1;
  684. ex->ZoomY *= 1.1;
  685. }
  686. laRedrawCurrentPanel();
  687. }
  688. return LA_FINISHED;
  689. }
  690. int OPMOD_CanvasZoom(laOperator *a, laEvent *e){
  691. laCanvasExtra *ex = a->This->EndInstance;
  692. laGeneralUiExtraData *uex = a->CustomData;
  693. if (e->Type == LA_L_MOUSE_UP || e->Type == LA_R_MOUSE_DOWN){
  694. laSetWindowCursor(LA_ARROW);
  695. return LA_FINISHED;
  696. }
  697. if (e->Type == LA_MOUSEMOVE){
  698. if (strArgumentMatch(a->ExtraInstructionsP, "axis", "x")){
  699. ex->ZoomX *= (1.0 - (e->x - uex->LastX) * MAIN.ZoomSpeed2D);
  700. }else{
  701. ex->ZoomX *= (1.0 - (e->x - uex->LastX) * MAIN.ZoomSpeed2D);
  702. ex->ZoomY *= (1.0 + (e->y - uex->LastY) * MAIN.ZoomSpeed2D);
  703. }
  704. uex->LastX = e->x;
  705. uex->LastY = e->y;
  706. laRedrawCurrentPanel();
  707. }
  708. return LA_RUNNING;
  709. }
  710. int OPMOD_CanvasMove(laOperator *a, laEvent *e){
  711. laCanvasExtra *ex = a->This->EndInstance;
  712. laGeneralUiExtraData *uex = a->CustomData;
  713. if (e->Type == LA_L_MOUSE_UP || e->Type == LA_R_MOUSE_DOWN){
  714. laSetWindowCursor(LA_ARROW);
  715. return LA_FINISHED;
  716. }
  717. if (e->Type == LA_MOUSEMOVE){
  718. if (strArgumentMatch(a->ExtraInstructionsP, "axis", "x")){
  719. ex->PanX -= (e->x - uex->LastX) * ex->ZoomX;
  720. }else{
  721. ex->PanX -= (e->x - uex->LastX) * ex->ZoomX;
  722. ex->PanY += (e->y - uex->LastY) * ex->ZoomY;
  723. }
  724. printf("%d %d\n",e->x,e->y);
  725. uex->LastX = e->x;
  726. uex->LastY = e->y;
  727. laRedrawCurrentPanel();
  728. }
  729. return LA_RUNNING;
  730. }
  731. int OPINV_CanvasClick(laOperator *a, laEvent *e){
  732. laCanvasExtra *ex = a->This->EndInstance;
  733. laGeneralUiExtraData *uex = a->CustomData;
  734. if (e->Type == LA_L_MOUSE_DOWN){
  735. ex->ClickedX = (e->x - (ex->ParentUi->R - ex->ParentUi->L) / 2 - ex->ParentUi->L) * ex->ZoomX + ex->PanX;
  736. ex->ClickedY = ((ex->ParentUi->B - ex->ParentUi->U) / 2 - e->y + ex->ParentUi->U) * ex->ZoomY + ex->PanY;
  737. laRedrawCurrentPanel();
  738. }
  739. return LA_FINISHED_PASS;
  740. }
  741. void la_RegisterViewerOperators(){
  742. laCreateOperatorType("LA_canvas_operator", "2D View UiItem Operator", "All Visual 2D View Operations Are Called From This Ui",
  743. 0, 0, OPEXT_3DOr2DViewUiItem, OPINV_3DOr2DViewUiItem, OPMOD_Canvas, L'🖦', LA_EXTRA_TO_PANEL | LA_ACTUATOR_SYSTEM | LA_ACTUATOR_HIDDEN);
  744. laCreateOperatorType("LA_3d_view_camera_zoom", "Camera Zoom", "Let View Camera Zoom In Or Out",
  745. OPCHK_Is3DViewExtra, 0, 0, OPINV_3DViewCameraZoom, 0, L'🔎', 0);
  746. laCreateOperatorType("LA_3d_view_camera_rotate", "Camera Rotate", "Let View Camera Rotate Along Local X/Y Axis",
  747. OPCHK_Is3DViewExtra, 0, OPEXT_3DOr2DViewAdjust, OPINV_3DOr2DViewAdjust, OPMOD_3DViewCameraRotate, L'🗘', LA_EXTRA_TO_PANEL);
  748. laCreateOperatorType("LA_3d_view_camera_move", "Camera Move", "Let View Camera Move Along Local X/Y Axis",
  749. OPCHK_Is3DViewExtra, 0, OPEXT_3DOr2DViewAdjust, OPINV_3DOr2DViewAdjust, OPMOD_3DViewCameraMove, L'🤚', LA_EXTRA_TO_PANEL);
  750. laCreateOperatorType("LA_2d_view_zoom", "2D Zoom", "Let View 2D Cavans Zoom In Or Out",
  751. OPCHK_Is2DViewExtra, 0, OPEXT_3DOr2DViewAdjust, OPINV_CanvasZoom, OPMOD_CanvasZoom, L'🔎', 0);
  752. laCreateOperatorType("LA_2d_view_move", "2D Move", "Let View 2D Cavans Move Along Local X/Y Axis",
  753. OPCHK_Is2DViewExtra, 0, OPEXT_3DOr2DViewAdjust, OPINV_3DOr2DViewAdjust, OPMOD_CanvasMove, L'🤚', LA_EXTRA_TO_PANEL);
  754. laCreateOperatorType("LA_2d_view_click", "2D Click", "2D Click",
  755. OPCHK_Is2DViewExtra, 0, OPEXT_3DOr2DViewAdjust, OPINV_CanvasClick, 0, L'🖦', LA_EXTRA_TO_PANEL);
  756. }
  757. void *tnsget_detached_FirstScene(void *UNUSED1, void *UNUSED2);
  758. void laset_CanvasSelectMode(laCanvasExtra* ex, int mode){
  759. ex->SelectMode=mode; laUiItem* ui=ex->ParentUi;
  760. tnsObject*root=ui?ui->PP.EndInstance:0; if(!root) return 0;
  761. if(root->Active){ if(root->Active->Type==TNS_OBJECT_MESH && ((tnsMeshObject*)root->Active)->Mode==TNS_MESH_EDIT_MODE){ tnsMMeshEnsureSelection(root->Active, mode); } }
  762. }
  763. void la_RegisterUiTypesViewerWidgets(){
  764. laPropContainer *pc = 0;
  765. laProp *p = 0;
  766. laKeyMapper *km;
  767. _LA_UI_CANVAS = la_RegisterUiType("LA_canvas_default", 0, "LA_canvas_operator", &_LA_THEME_2D_VIEW, la_CanvasDraw, la_CanvasGetHeight, la_CanvasUiInit, la_CanvasUiDestroy);
  768. _LA_UI_CANVAS->Tag = LA_UI_TAG_NEED_REBUILD;
  769. laCanvasTemplate* ct=laRegisterCanvasTemplate("la_CanvasDrawTexture", "tns_texture", la_CanvasDrawTexture, la_CanvasDrawOverlay, la_CanvasInit, la_CanvasDestroy);
  770. pc = laCanvasHasExtraProps(ct,sizeof(laCanvasExtra),2);{
  771. _LA_PROP_2D_EXTRA = pc;
  772. laAddIntProperty(pc, "height_coeff", "Ui Height", "Ui Height Coefficiency Entry", 0, 0, "Rows", 100, -100, 1, 0, 0, offsetof(laCanvasExtra, HeightCoeff), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
  773. p = laAddEnumProperty(pc, "draw_image_alpha", "Draw Image Alpha", "Draw Grid Background On Alpha<1", 0, 0, 0, 0, 0,
  774. offsetof(laCanvasExtra, ImageDrawAlpha), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);{
  775. laAddEnumItem(p, "null", "null", "Don't Draw Grid", 0);
  776. laAddEnumItem(p, "normal", "normal", "Use Normal Color", 0);
  777. laAddEnumItem(p, "bright", "bright", "Draw Alpha Grid Using Very Bright White Color", 0);
  778. }
  779. p = laAddEnumProperty(pc, "draw_image_border", "Draw Image Border", "Draw Image Border Using Same Color As Ui Item", 0, 0, 0, 0, 0,
  780. offsetof(laCanvasExtra, ImageDrawBorder), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);{
  781. laAddEnumItem(p, "no", "No", "Don't Draw Border", L'☐');
  782. laAddEnumItem(p, "yes", "Yes", "Draw Border", L'☑');
  783. }
  784. p = laAddEnumProperty(pc, "adaptive_line_width", "Adaptive Line Width", "glLineWith() will follow 2dview zooming", 0, 0, 0, 0, 0,
  785. offsetof(laCanvasExtra, AdaptiveLineWidth), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);{
  786. laAddEnumItem(p, "constant", "Constant", "Don't change line width", L'☐');
  787. laAddEnumItem(p, "adaptive", "Adaptive", "Adaptive Line Width", L'☑');
  788. }
  789. p = laAddEnumProperty(pc, "line_width_warning", "Line Width Warnning", "Show whether line width is acceptable by hardware", 0, 0, 0, 0, 0,
  790. offsetof(laCanvasExtra, LineWidthWarning), 0, 0, 0, 0, 0, 0, 0, 0, 0, LA_READ_ONLY);{
  791. laAddEnumItem(p, "acceptable", "Acceptable", "Line width is acceptable by graphic hadware", L'✔');
  792. laAddEnumItem(p, "too_wide", "Too Wide", "Line width is too wide for graphic hadware", L'❌');
  793. laAddEnumItem(p, "too_thin", "Too Thin", "Line width is too thin for graphic hadware", L'❌');
  794. }
  795. p = laAddEnumProperty(pc, "clear_background", "Clear Background", "Clear Background", 0, 0, 0, 0, 0, offsetof(laCanvasExtra, ClearBackground), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);{
  796. laAddEnumItem(p, "no", "No", "Don't Clear Background", L'🌔');
  797. laAddEnumItem(p, "yes", "Yes", "Clear Background", L'🌑');
  798. }
  799. p = laAddEnumProperty(pc, "draw_cursor", "Show Cursor", "Show Cursor", 0, 0, 0, 0, 0, offsetof(laCanvasExtra, DrawCursor), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);{
  800. laAddEnumItem(p, "no", "No", "Don't draw cursor", L'🖦');
  801. laAddEnumItem(p, "yes", "Yes", "Draw cursor", L'➕');
  802. }
  803. laAddFloatProperty(pc, "pan", "Pan", "Pan On X,Y Axis", 0, "X,Y", "px", 0, 0, 1, 0, 0, offsetof(laCanvasExtra, PanX), 0, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0);
  804. laAddFloatProperty(pc, "zoom", "Zoom", "Zoom Factor On X,Y Axis", 0, "X,Y", 0, 0, 0, 0.01, 1, 0, offsetof(laCanvasExtra, ZoomX), 0, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0);
  805. //laAddSubGroup(pc, "Template", "Template Used To Draw 2D Stuff", "la_2d_view_template",0, 0, 0, -offsetof(laCanvasExtra, Template), 0, 0, 0, 0, 0, 0, 0, 0, LA_UDF_REFER);
  806. laAddOperatorProperty(pc, "zoom", "Zoom", "Zoom 2D Canvans", "LA_2d_view_zoom", L'🔎', 0);
  807. laAddOperatorProperty(pc, "move", "Move", "Move 2D Canvans", "LA_2d_view_move", L'🤚', 0);
  808. laAddOperatorProperty(pc, "click", "Click", "Click On 2D Canvans", "LA_2d_view_click", L'🤚', 0);
  809. laAddSubGroup(pc, "parent_ui", "Parent UI", "The Ui That Holds The Viewer", "ui_item",0, 0, 0, offsetof(laCanvasExtra, ParentUi), 0, 0, 0, 0, 0, 0, 0, LA_UDF_REFER);
  810. }
  811. km = &ct->KeyMapper;
  812. laAssignNewKey(km, 0, "LA_2d_view_zoom", LA_KM_SEL_UI_EXTRA, 0, LA_MOUSE_WHEEL_DOWN, 0, "direction=out");
  813. laAssignNewKey(km, 0, "LA_2d_view_zoom", LA_KM_SEL_UI_EXTRA, 0, LA_MOUSE_WHEEL_UP, 0, "direction=in");
  814. laAssignNewKey(km, 0, "LA_2d_view_move", LA_KM_SEL_UI_EXTRA, LA_KEY_ALT, LA_L_MOUSE_DOWN, 0, 0);
  815. laAssignNewKey(km, 0, "LA_2d_view_click", LA_KM_SEL_UI_EXTRA, 0, LA_L_MOUSE_DOWN, 0, 0);
  816. ct=laRegisterCanvasTemplate("la_3DView", "tns_object", la_RootObjectDraw, la_RootObjectDrawOverlay, la_3DViewInit, la_3DViewDestroy);
  817. pc = laCanvasHasExtraProps(ct, sizeof(laCanvasExtra), 2);{
  818. _LA_PROP_3D_EXTRA = pc;
  819. laAddIntProperty(pc, "grid_size", "Grid Size", "Floor Grid Size Per Cell", 0, 0, "Unit", 100, 1, 1, 10, 0, offsetof(laCanvasExtra, GridSize), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
  820. laAddIntProperty(pc, "grid_span", "Grid Span", "How Many Grids Are Drawn", 0, 0, 0, 25, 1, 1, 10, 0, offsetof(laCanvasExtra, GridSpan), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
  821. p = laAddEnumProperty(pc, "show_axis", "Show Axis", "Show Global X,Y,Z Axis", LA_WIDGET_ENUM_CYCLE, "X,Y,Z", 0, 0, 0, offsetof(laCanvasExtra, ShowAxis), 0, 0, 3, 0, 0, 0, 0, 0, 0, 0);{
  822. laAddEnumItem(p, "HIDDEN", "Hidden", "Current Axis Is Hidden", L'🌔');
  823. laAddEnumItem(p, "SHOWN", "Shown", "Current Axis Is Shown", L'🌑');
  824. }
  825. p = laAddEnumProperty(pc, "show_floor_grid", "Show Floor Grid", "Show Floor Grid", 0, 0, 0, 0, 0, offsetof(laCanvasExtra, ShowFloorGrid), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);{
  826. laAddEnumItem(p, "HIDDEN", "Hidden", "Current Axis Is Hidden", L'🌔');
  827. laAddEnumItem(p, "SHOWN", "Shown", "Current Axis Is Shown", L'🌑');
  828. }
  829. p = laAddEnumProperty(pc, "select_mode", "Select Mode", "Select by vertices or edges", 0, 0, 0, 0, 0, offsetof(laCanvasExtra, SelectMode), 0, laset_CanvasSelectMode, 0, 0, 0, 0, 0, 0, 0, 0);{
  830. laAddEnumItemAs(p, "VERTS", "Verts", "Select by vertices", LA_CANVAS_SELECT_MODE_VERTS,0);
  831. laAddEnumItemAs(p, "EDGES", "Edges", "Select by edges", LA_CANVAS_SELECT_MODE_EDGES,0);
  832. }
  833. p = laAddEnumProperty(pc, "select_through", "Select Through", "Select through stuff", LA_WIDGET_ENUM_HIGHLIGHT, 0, 0, 0, 0, offsetof(laCanvasExtra, SelectThrough), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);{
  834. laAddEnumItemAs(p, "OFF", "Off", "Don't select through stuff", LA_CANVAS_SELECT_THROUGH_OFF,0);
  835. laAddEnumItemAs(p, "ON", "On", "Select through stuff", LA_CANVAS_SELECT_THROUGH_ON,0);
  836. }
  837. laAddIntProperty(pc, "height_coeff", "Ui Height", "Ui Height Coefficiency Entry", 0, 0, "Rows", 100, -100, 1, 0, 0, offsetof(laCanvasExtra, HeightCoeff), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
  838. laAddOperatorProperty(pc, "zoom", "Zoom", "Zoom Viewing Camera", "LA_3d_view_camera_zoom", L'🔎', 0);
  839. laAddOperatorProperty(pc, "rotate", "Rotate", "Rotate Viewing Camera", "LA_3d_view_camera_rotate", L'🗘', 0);
  840. laAddOperatorProperty(pc, "move", "Move", "Move Viewing Camera", "LA_3d_view_camera_move", L'🤚', 0);
  841. //laAddSubGroup(pc, "Viewing Camera", "Unique Camera That Is Used To Draw 3D Viewport", "tns_object",0, 0, 0, -offsetof(laCanvasExtra, ViewingCamera), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
  842. laAddSubGroup(pc, "parent_ui", "Parent Ui", "Parent Ui (Mostly Used To Determin Viewport Size)", "ui_item",0, 0, 0, offsetof(laCanvasExtra, ParentUi), 0, 0, 0, 0, 0, 0, 0, LA_UDF_REFER);
  843. laAddOperatorProperty(pc, "align_camera_to_view", "Align Active Camera To View", "Align Active Camera To View", "TNS_align_camera_to_view", L'🎥', 0);
  844. laAddOperatorProperty(pc, "align_view_to_camera", "Align View To Active Camera", "Align View To Active Camera", "TNS_align_view_to_camera", L'🎥', 0);
  845. laAddOperatorProperty(pc, "_this_M_delete", "Delete", "Delete parts of mesh", "M_delete", 0, 0);
  846. laAddOperatorProperty(pc, "_this_M_make_parent", "Make parent", "Parent objects to active objects or unparent selected ones", "M_make_parent", 0, 0);
  847. laAddOperatorProperty(pc, "_this_M_unparent", "Unparent", "Unparent selected objects", "M_unparent", 0, 0);
  848. laAddOperatorProperty(pc, "_this_M_add", "Add", "Add objects/primitives", "M_add", 0, 0);
  849. laAddOperatorProperty(pc, "_this_M_grab", "Grab", "Grab things and move around", "M_grab", 0, 0);
  850. laAddOperatorProperty(pc, "_this_M_scale", "Scale", "Scale selected things", "M_scale", 0, 0);
  851. laAddOperatorProperty(pc, "_this_M_rotate", "Rotate", "Rotation selected things", "M_rotate", 0, 0);
  852. laAddOperatorProperty(pc, "_this_M_clear_transformations", "Clear Transformations", "Clear object transformation values", "M_clear_transformations", 0, 0);
  853. laAddOperatorProperty(pc, "_this_M_extrude", "Extrude", "Extrude parts of the mesh", "M_extrude", 0, 0);
  854. laAddOperatorProperty(pc, "_this_M_delete", "Delete", "Delete parts of the mesh", "M_delete", 0, 0);
  855. laAddOperatorProperty(pc, "_this_M_make", "Make", "Make mesh primitive from selected ones", "M_make", 0, 0);
  856. laAddOperatorProperty(pc, "_this_M_subdiv", "Subdiv", "Subdivide edges", "M_subdiv", 0, 0);
  857. laAddOperatorProperty(pc, "_this_M_add", "Add", "Add objects/primitives", "M_add", 0, 0);
  858. laAddOperatorProperty(pc, "_this_M_separate", "Separate", "Separate mesh parts", "M_separate", 0, 0);
  859. laAddOperatorProperty(pc, "_this_M_combine", "Combine", "Combine mesh objects", "M_combine", 0, 0);
  860. laAddOperatorProperty(pc, "_this_M_duplicate", "Duplicate", "Duplicate objects", "M_duplicate", 0, 0);
  861. laAddOperatorProperty(pc, "_this_M_recalculate_normals", "Recalculate Normals", "Recalculate normals", "M_recalculate_normals", 0, 0);
  862. }
  863. km = &ct->KeyMapper;
  864. laAssignNewKey(km, 0, "LA_3d_view_camera_zoom", LA_KM_SEL_UI_EXTRA, 0, LA_MOUSE_WHEEL_DOWN, 0, "direction=out");
  865. laAssignNewKey(km, 0, "LA_3d_view_camera_zoom", LA_KM_SEL_UI_EXTRA, 0, LA_MOUSE_WHEEL_UP, 0, "direction=in");
  866. laAssignNewKey(km, 0, "LA_3d_view_camera_rotate", LA_KM_SEL_UI_EXTRA, LA_KEY_ALT, LA_L_MOUSE_DOWN, 0, 0);
  867. laAssignNewKey(km, 0, "LA_3d_view_camera_rotate", LA_KM_SEL_UI_EXTRA, 0, LA_M_MOUSE_DOWN, 0, 0);
  868. laAssignNewKey(km, 0, "LA_3d_view_camera_move", LA_KM_SEL_UI_EXTRA, LA_KEY_SHIFT, LA_M_MOUSE_DOWN, 0, 0);
  869. laAssignNewKey(km, 0, "LA_3d_view_camera_move", LA_KM_SEL_UI_EXTRA, LA_KEY_SHIFT | LA_KEY_ALT, LA_L_MOUSE_DOWN, 0, 0);
  870. laAssignNewKey(km, 0, "TNS_align_camera_to_view", LA_KM_SEL_UI_EXTRA, LA_KEY_ALT | LA_KEY_SHIFT, LA_R_MOUSE_DOWN, 0, 0);
  871. laAssignNewKey(km, 0, "TNS_align_view_to_camera", LA_KM_SEL_UI_EXTRA, 0, LA_KEY_DOWN, '0', 0);
  872. laAssignNewKey(km, 0, "M_toggle_edit_mode", LA_KM_SEL_UI_EXTRA, 0, LA_KEY_DOWN, LA_KEY_TAB, 0);
  873. laAssignNewKey(km, 0, "M_select", LA_KM_SEL_UI_EXTRA, LA_KEY_ALT, LA_R_MOUSE_DOWN, 0, 0);
  874. laAssignNewKey(km, 0, "M_select", LA_KM_SEL_UI_EXTRA, LA_KEY_SHIFT, LA_R_MOUSE_DOWN, 0, 0);
  875. laAssignNewKey(km, 0, "M_select", LA_KM_SEL_UI_EXTRA, 0, LA_R_MOUSE_DOWN, 0, 0);
  876. laAssignNewKey(km, 0, "M_select", LA_KM_SEL_UI_EXTRA, 0, LA_KEY_DOWN, 'b', "mode=box;");
  877. laAssignNewKey(km, 0, "M_select", LA_KM_SEL_UI_EXTRA, 0, LA_KEY_DOWN, 'a', "mode=toggle;");
  878. laAssignNewKey(km, 0, "M_grab", LA_KM_SEL_UI_EXTRA, 0, LA_KEY_DOWN, 'g', 0);
  879. laAssignNewKey(km, 0, "M_scale", LA_KM_SEL_UI_EXTRA, 0, LA_KEY_DOWN, 's', 0);
  880. laAssignNewKey(km, 0, "M_rotate", LA_KM_SEL_UI_EXTRA, 0, LA_KEY_DOWN, 'r', 0);
  881. laAssignNewKey(km, 0, "M_make_parent", LA_KM_SEL_UI_EXTRA, LA_KEY_CTRL, LA_KEY_DOWN, 'p', 0);
  882. laAssignNewKey(km, 0, "M_unparent", LA_KM_SEL_UI_EXTRA, LA_KEY_ALT, LA_KEY_DOWN, 'p', 0);
  883. laAssignNewKey(km, 0, "M_clear_transformations", LA_KM_SEL_UI_EXTRA, LA_KEY_ALT, LA_KEY_DOWN, 'g', "location=true;text=Clear Location;");
  884. laAssignNewKey(km, 0, "M_clear_transformations", LA_KM_SEL_UI_EXTRA, LA_KEY_ALT, LA_KEY_DOWN, 'r', "rotation=true;text=Clear Rotation;");
  885. laAssignNewKey(km, 0, "M_clear_transformations", LA_KM_SEL_UI_EXTRA, LA_KEY_ALT, LA_KEY_DOWN, 's', "scale=true;text=Clear Scale;");
  886. laAssignNewKey(km, 0, "M_extrude", LA_KM_SEL_UI_EXTRA, 0, LA_KEY_DOWN, 'e', 0);
  887. laAssignNewKey(km, 0, "M_extrude", LA_KM_SEL_UI_EXTRA, LA_KEY_SHIFT, LA_KEY_DOWN, 'd', "duplicate_only=true;text=Duplicate;");
  888. laAssignNewKey(km, 0, "M_delete", LA_KM_SEL_UI_EXTRA, 0, LA_KEY_DOWN, 'x', 0);
  889. laAssignNewKey(km, 0, "M_make", LA_KM_SEL_UI_EXTRA, 0, LA_KEY_DOWN, 'f', 0);
  890. laAssignNewKey(km, 0, "M_subdiv", LA_KM_SEL_UI_EXTRA, 0, LA_KEY_DOWN, 'w', 0);
  891. laAssignNewKey(km, 0, "M_add", LA_KM_SEL_UI_EXTRA, LA_KEY_SHIFT, LA_KEY_DOWN, 'a', 0);
  892. laAssignNewKey(km, 0, "M_separate", LA_KM_SEL_UI_EXTRA, 0, LA_KEY_DOWN, 'p', 0);
  893. laAssignNewKey(km, 0, "M_combine", LA_KM_SEL_UI_EXTRA, LA_KEY_CTRL, LA_KEY_DOWN, 'j', 0);
  894. laAssignNewKey(km, 0, "M_duplicate", LA_KM_SEL_UI_EXTRA, LA_KEY_SHIFT, LA_KEY_DOWN, 'd', 0);
  895. laAssignNewKey(km, 0, "M_recalculate_normals", LA_KM_SEL_UI_EXTRA, LA_KEY_CTRL, LA_KEY_DOWN, 'n', 0);
  896. }