mainWind.threejs.ts 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646
  1. import * as THREE from 'three';
  2. import { CSS3DObject } from 'three/examples/jsm/renderers/CSS3DRenderer.js';
  3. import Smoke from '/@/views/vent/comment/threejs/Smoke';
  4. import SmokePath from '/@/views/vent/comment/threejs/SmokePath';
  5. import { getTextCanvas, renderVideo } from '/@/utils/threejs/util';
  6. import gsap from 'gsap';
  7. import { Mesh } from 'three';
  8. import { resolve } from 'path';
  9. import Matching from '@zxcvbn-ts/core/dist/Matching';
  10. class mainWindRect {
  11. model;
  12. modelName = 'main';
  13. group: THREE.Group | null = null; // 主通风机场景
  14. motorGroup1: THREE.Group | null = null; //电机
  15. motorGroup2: THREE.Group | null = null; //电机
  16. gearFront = {
  17. gear1: null, //扇叶
  18. gear2: null, //扇叶
  19. gear1Direction: -1,
  20. gear2Direction: 1,
  21. gearFrameId: undefined,
  22. gearRotateFactor: 0.5,
  23. endGearRotateFactor: 3,
  24. };
  25. gearBack = {
  26. gear1: null, //扇叶
  27. gear2: null, //扇叶
  28. gear1Direction: -1, // 扇叶转动方向
  29. gear2Direction: 1, // 扇叶转动方向
  30. gearFrameId: undefined,
  31. gearRotateFactor: 0.5, // 扇叶转动因素
  32. endGearRotateFactor: 3, // 扇叶最终转动速度因素
  33. };
  34. // smoke;
  35. frontSmoke; // 前面风流对象
  36. backSmoke; // 后面风流对象
  37. player1; // 视频播放器
  38. playerStartClickTime1 = new Date().getTime();
  39. frontWindowGroup;
  40. backWindowGroup;
  41. windowAngle = 0;
  42. constructor(model, playerVal1) {
  43. this.model = model;
  44. this.player1 = playerVal1;
  45. }
  46. // 添加 cssObject
  47. addCssText() {
  48. if (!this.group) {
  49. return;
  50. }
  51. if (!this.group.getObjectByName('monitorText1')) {
  52. const element = document.getElementById('inputBox') as HTMLElement;
  53. const mainCSS3D = new CSS3DObject(element);
  54. mainCSS3D.name = 'monitorText1';
  55. mainCSS3D.scale.set(0.08, 0.08, 0.08);
  56. mainCSS3D.position.set(23.78, 18.18, -6.85);
  57. mainCSS3D.lookAt(23.78, 20, -2.85);
  58. this.group.add(mainCSS3D);
  59. }
  60. if (!this.group.getObjectByName('monitorText2')) {
  61. const element = document.getElementById('inputBox1') as HTMLElement;
  62. const mainCSS3D = new CSS3DObject(element);
  63. mainCSS3D.name = 'monitorText2';
  64. mainCSS3D.scale.set(0.08, 0.08, 0.08);
  65. mainCSS3D.position.set(23.78, 18.18, 16.82);
  66. mainCSS3D.lookAt(23.78, 20, 20.82);
  67. this.group.add(mainCSS3D);
  68. }
  69. }
  70. addEcharts() {
  71. const echartsBox = document.getElementById('fan-echarts');
  72. if (echartsBox) {
  73. const canvasObj = echartsBox.getElementsByTagName('canvas')[0];
  74. // 将canvas 纹理转换为材质
  75. const echartsMap = new THREE.CanvasTexture(canvasObj); // 关键一步
  76. const echartsMaterial = new THREE.MeshBasicMaterial({
  77. map: echartsMap, // 设置纹理贴图
  78. transparent: true,
  79. side: THREE.FrontSide, // 这里是双面渲染的意思
  80. });
  81. echartsMaterial.blending = THREE.CustomBlending;
  82. const monitorPlane = this.group?.getObjectByName('monitorEcharts');
  83. if (monitorPlane) {
  84. monitorPlane.material = echartsMaterial;
  85. } else {
  86. const planeGeometry = new THREE.PlaneGeometry(17.6, 9.9); // 平面3维几何体PlaneGeometry
  87. const planeMesh = new THREE.Mesh(planeGeometry, echartsMaterial);
  88. planeMesh.name = 'monitorEcharts';
  89. planeMesh.scale.set(1, 1, 1);
  90. planeMesh.position.set(-21.98, 9.027, -21.79);
  91. this.group?.add(planeMesh);
  92. }
  93. }
  94. }
  95. initAnimation() {}
  96. startAnimation() {}
  97. /* 更新动画 */
  98. render() {
  99. if (!this.model) {
  100. return;
  101. }
  102. }
  103. /* 点击风窗,风窗全屏 */
  104. mousedownModel(intersects: THREE.Intersection<THREE.Object3D<THREE.Event>>[]) {
  105. // 判断是否点击到视频
  106. intersects.find((intersect) => {
  107. const mesh = intersect.object;
  108. if (mesh.name === 'player1') {
  109. if (new Date().getTime() - this.playerStartClickTime1 < 400) {
  110. // 双击,视频放大
  111. if (this.player1) {
  112. this.player1.requestFullscreen();
  113. }
  114. }
  115. this.playerStartClickTime1 = new Date().getTime();
  116. return true;
  117. }
  118. return false;
  119. });
  120. }
  121. mouseUpModel() {}
  122. /* 初始化气流 */
  123. initSmoke() {
  124. const frontPath = new THREE.CatmullRomCurve3([
  125. new THREE.Vector3(7.441, 0.806, 3.614),
  126. new THREE.Vector3(11.087, 0.806, 3.614),
  127. new THREE.Vector3(22.851, 0.806, 3.614),
  128. new THREE.Vector3(28.733, 0.806, 3.614),
  129. new THREE.Vector3(34.615, 0.806, 3.614),
  130. new THREE.Vector3(39.802, 0.806, 3.614),
  131. new THREE.Vector3(41.583, 1.485, 3.614),
  132. new THREE.Vector3(42.741, 3.364, 3.614),
  133. new THREE.Vector3(42.741, 7.998, 3.614),
  134. new THREE.Vector3(42.741, 12.633, 3.614),
  135. new THREE.Vector3(42.741, 17.267, 3.614),
  136. ]);
  137. this.frontSmoke = new SmokePath('/model/img/texture-smoke.png', '#ffffff', 0, 1, 0.7, 300, frontPath);
  138. this.frontSmoke.setPoints();
  139. this.frontSmoke.pointsMesh.position.set(0.33, 2.8, 8.04);
  140. this.group?.add(this.frontSmoke.pointsMesh);
  141. const backSmoke = new THREE.CatmullRomCurve3([
  142. new THREE.Vector3(7.441, 0.806, 3.614),
  143. new THREE.Vector3(11.087, 0.806, 3.614),
  144. new THREE.Vector3(22.851, 0.806, 3.614),
  145. new THREE.Vector3(28.733, 0.806, 3.614),
  146. new THREE.Vector3(34.615, 0.806, 3.614),
  147. new THREE.Vector3(39.802, 0.806, 3.614),
  148. new THREE.Vector3(41.583, 1.485, 3.614),
  149. new THREE.Vector3(42.741, 3.364, 3.614),
  150. new THREE.Vector3(42.741, 7.998, 3.614),
  151. new THREE.Vector3(42.741, 12.633, 3.614),
  152. new THREE.Vector3(42.741, 17.267, 3.614),
  153. ]);
  154. this.backSmoke = new SmokePath('/model/img/texture-smoke.png', '#ffffff', 0, 1, 0.7, 300, backSmoke);
  155. this.backSmoke.setPoints();
  156. this.backSmoke.pointsMesh.position.set(0.65, 2.77, -7.38);
  157. this.group?.add(this.backSmoke.pointsMesh);
  158. }
  159. async setDeviceFrequency(deviceType, state, frequencyVal?) {
  160. // 调节频率
  161. if (frequencyVal) {
  162. this.resetSmokeParam(deviceType, frequencyVal, 0);
  163. }
  164. this.openOrCloseValve(deviceType, state, 0);
  165. this.startGearAnimation(deviceType, state, '', 0);
  166. if (deviceType === 'front') {
  167. this.frontSmoke.startSmoke();
  168. } else {
  169. this.backSmoke.startSmoke();
  170. }
  171. setTimeout(() => {
  172. this.lookMotor(deviceType, state, 10);
  173. }, 2000);
  174. }
  175. async openDevice(deviceType, smokeDirection, frequencyVal, duration?) {
  176. if (smokeDirection) {
  177. this.setSmokeDirection(deviceType, smokeDirection);
  178. }
  179. let smoke;
  180. if (deviceType === 'front') {
  181. smoke = this.frontSmoke;
  182. } else {
  183. smoke = this.backSmoke;
  184. }
  185. if (!smoke.frameId) {
  186. await this.lookMotor(deviceType, 'open', duration);
  187. await this.openOrCloseValve(deviceType, 'open', duration);
  188. this.startGearAnimation(deviceType, 'open', smokeDirection, frequencyVal, duration);
  189. smoke.startSmoke(duration);
  190. }
  191. }
  192. async closeDevice(deviceType) {
  193. let smoke;
  194. if (deviceType === 'front') {
  195. smoke = this.frontSmoke;
  196. } else if (deviceType === 'back') {
  197. smoke = this.backSmoke;
  198. }
  199. if (smoke && smoke.frameId) {
  200. smoke.stopSmoke();
  201. await this.openOrCloseValve(deviceType, 'close');
  202. this.startGearAnimation(deviceType, 'close', '', null);
  203. await this.lookMotor(deviceType, 'close');
  204. }
  205. }
  206. async setSmokeDirection(deviceType, smokeDirection) {
  207. const _this = this;
  208. const tubPositivePath = [
  209. {
  210. path0: new THREE.Vector3(7.441, 0.806, 3.614),
  211. path1: new THREE.Vector3(41.583, 1.485, 3.614),
  212. isSpread: false,
  213. spreadDirection: 0, //
  214. },
  215. {
  216. path0: new THREE.Vector3(41.583, 1.485, 3.614),
  217. path1: new THREE.Vector3(42.741, 5.364, 3.614),
  218. isSpread: false,
  219. spreadDirection: 0,
  220. },
  221. {
  222. path0: new THREE.Vector3(42.741, 5.364, 3.614),
  223. path1: new THREE.Vector3(44.741, 17.267, 3.614),
  224. isSpread: true,
  225. spreadDirection: 1, // 1是由小变大,-1是由大变小
  226. },
  227. ];
  228. const tubInversePath = [
  229. {
  230. path0: new THREE.Vector3(44.741, 17.267, 3.614),
  231. path1: new THREE.Vector3(42.741, 5.364, 3.614),
  232. isSpread: true,
  233. spreadDirection: -1, //
  234. },
  235. {
  236. path0: new THREE.Vector3(42.741, 5.364, 3.614),
  237. path1: new THREE.Vector3(41.583, 1.485, 3.614),
  238. isSpread: false,
  239. spreadDirection: 0, //
  240. },
  241. {
  242. path0: new THREE.Vector3(41.583, 1.485, 3.614),
  243. path1: new THREE.Vector3(7.441, 0.806, 3.614),
  244. isSpread: false,
  245. spreadDirection: 0, // 1是由小变大,-1是由大变小
  246. },
  247. ];
  248. if (deviceType === 'front') {
  249. switch (smokeDirection) {
  250. case 'tubPositivePath': // 风筒正
  251. _this.frontSmoke.setPath(tubPositivePath);
  252. break;
  253. case 'tubInversePath': // 风筒反
  254. _this.frontSmoke.setPath(tubInversePath);
  255. break;
  256. }
  257. }
  258. }
  259. /* 播放气流动画 */
  260. /**
  261. *
  262. * @param controlType // 设备控制类型
  263. * @param deviceType //前后风机
  264. * @param frequencyVal // 风机运行频率
  265. * @param state // 打开、关闭状态
  266. */
  267. async playSmoke(controlType, deviceType, frequencyVal, state, smokeDirection) {
  268. if (frequencyVal) {
  269. this.resetSmokeParam(deviceType, frequencyVal);
  270. }
  271. if (controlType === 'startSmoke') {
  272. if (state === 'stop') {
  273. await this.closeDevice(deviceType);
  274. } else {
  275. // 开启时需要设置方向
  276. await this.openDevice(deviceType, smokeDirection, frequencyVal);
  277. }
  278. } else if (controlType === 'changeDirection') {
  279. // 改变扇叶转动方向、反风
  280. this.startGearAnimation(deviceType, 'changeDirection', smokeDirection, frequencyVal);
  281. let smoke;
  282. if (deviceType === 'front') {
  283. smoke = this.frontSmoke;
  284. } else {
  285. smoke = this.backSmoke;
  286. }
  287. if (smoke && smoke.frameId) {
  288. await smoke.stopSmoke();
  289. await this.setSmokeDirection(deviceType, smokeDirection);
  290. smoke.startSmoke();
  291. }
  292. } else if (controlType === 'frequency') {
  293. this.startGearAnimation(deviceType, 'frequency', smokeDirection, frequencyVal);
  294. } else if (controlType === 'initiatePlay') {
  295. this.openDevice(deviceType, smokeDirection, frequencyVal, 0);
  296. }
  297. }
  298. /* 打开或关闭蝶阀 */
  299. openOrCloseValve(deviceType, flag, duration = 3) {
  300. return new Promise((resolve) => {
  301. let diefa;
  302. if (deviceType == 'front') {
  303. diefa = this.group?.getObjectByName('butterfly_valve001') as THREE.Mesh;
  304. } else {
  305. diefa = this.group?.getObjectByName('butterfly_valve002') as THREE.Mesh;
  306. }
  307. let rotationY;
  308. if (flag == 'open') {
  309. rotationY = 0;
  310. } else {
  311. rotationY = Math.PI / 2;
  312. }
  313. gsap.to(diefa.rotation, {
  314. y: rotationY,
  315. duration: duration,
  316. ease: 'none',
  317. onComplete: function () {
  318. resolve(null);
  319. },
  320. });
  321. });
  322. }
  323. /* 风流调频, 范围1-50 */
  324. // opacityFactor (0.4 300)
  325. // life 最小 300, 最大 50
  326. // speedFactor 最大0, 最小100
  327. resetSmokeParam(deviceType, frequency, duration = 5) {
  328. if (frequency < 1) frequency = 1;
  329. if (frequency > 50) frequency = 50;
  330. let smoke;
  331. if (deviceType === 'front') {
  332. smoke = this.frontSmoke;
  333. } else {
  334. smoke = this.backSmoke;
  335. }
  336. const opacityFactor = (frequency / 50) * 0.8;
  337. duration = (Number(Math.abs(smoke.opacityFactor - opacityFactor).toFixed(1)) / 0.8) * 5;
  338. const life = (-250 / 50) * frequency + 300;
  339. gsap.to(smoke, {
  340. opacityFactor: opacityFactor,
  341. life: life,
  342. duration: duration,
  343. ease: 'easeInCirc',
  344. overwrite: true,
  345. });
  346. }
  347. /* 显示电机 */
  348. lookMotor(deviceType, flag, duration = 5) {
  349. return new Promise((resolve) => {
  350. let mesh, motorGroup;
  351. if (deviceType == 'front') {
  352. mesh = this.group?.getObjectByName('TWO02');
  353. motorGroup = this.motorGroup2;
  354. } else {
  355. mesh = this.group?.getObjectByName('ONE00');
  356. motorGroup = this.motorGroup1;
  357. }
  358. if (flag == 'open') {
  359. motorGroup.visible = true;
  360. mesh.material.transparent = true;
  361. mesh.material.depthWrite = false;
  362. mesh.material.depthTest = true;
  363. mesh.material.sizeAttenuation = true;
  364. gsap.to(mesh.material, {
  365. opacity: 0.015,
  366. duration: duration,
  367. overwrite: true,
  368. onComplete: function () {
  369. resolve(null);
  370. },
  371. });
  372. } else {
  373. gsap.to(mesh.material, {
  374. opacity: 1,
  375. duration: 1,
  376. overwrite: true,
  377. onComplete: function () {
  378. // resolve(null);
  379. },
  380. });
  381. }
  382. });
  383. }
  384. /* 齿轮转动动画 1 - 50 最大3 */
  385. startGearAnimation(deviceType, flag, smokeDirection, frequencyVal, duration = 8) {
  386. let gearObj, gearDirection;
  387. if (deviceType === 'front') {
  388. gearObj = this.gearFront;
  389. } else {
  390. gearObj = this.gearBack;
  391. }
  392. if (smokeDirection === 'tubPositivePath') {
  393. gearDirection = 1;
  394. } else if (smokeDirection === 'tubInversePath') {
  395. gearDirection = -1;
  396. }
  397. if (frequencyVal) {
  398. const endGearRotateFactor = (3 / 50) * frequencyVal;
  399. duration = (8 / 3) * Math.abs(gearObj.endGearRotateFactor - endGearRotateFactor);
  400. gearObj.endGearRotateFactor = endGearRotateFactor;
  401. }
  402. const gearAnimation = () => {
  403. gsap.to(gearObj, {
  404. gearRotateFactor: gearObj.endGearRotateFactor,
  405. duration: duration,
  406. ease: 'easeInCubic',
  407. repeat: 0,
  408. overwrite: true,
  409. });
  410. const clock = new THREE.Clock(); // 时钟
  411. const h = () => {
  412. gearObj.gearFrameId = requestAnimationFrame(h);
  413. const dt = clock.getDelta();
  414. gearObj.gear1.rotation.x += dt * gearObj.gearRotateFactor * gearObj.gear1Direction;
  415. gearObj.gear2.rotation.x += dt * gearObj.gearRotateFactor * gearObj.gear2Direction;
  416. };
  417. gearObj.gearFrameId = requestAnimationFrame(h);
  418. };
  419. if (flag === 'changeDirection') {
  420. if (gearDirection == -1 * gearObj.gear1Direction) {
  421. // 齿轮正在转,需要停止后再反方向转
  422. gsap.to(gearObj, {
  423. gearRotateFactor: 0,
  424. duration: duration,
  425. ease: 'easeInCubic',
  426. repeat: 0,
  427. onComplete: function () {
  428. window.cancelAnimationFrame(gearObj.gearFrameId);
  429. gearObj.gearFrameId = undefined;
  430. gearObj.gear1Direction = -1 * gearObj.gear1Direction;
  431. gearObj.gear2Direction = -1 * gearObj.gear2Direction;
  432. gearAnimation();
  433. },
  434. });
  435. }
  436. } else if (flag === 'open') {
  437. gearObj.gear1Direction = gearDirection;
  438. gearAnimation();
  439. } else if (flag === 'close') {
  440. gsap.to(gearObj, {
  441. gearRotateFactor: 0,
  442. duration: duration,
  443. ease: 'easeInCubic',
  444. repeat: 0,
  445. overwrite: true,
  446. onComplete: function () {
  447. window.cancelAnimationFrame(gearObj.gearFrameId);
  448. gearObj.gearFrameId = undefined;
  449. },
  450. });
  451. } else if (flag === 'frequency') {
  452. gsap.to(gearObj, {
  453. gearRotateFactor: gearObj.endGearRotateFactor,
  454. duration: duration,
  455. ease: 'easeInCubic',
  456. repeat: 0,
  457. overwrite: true,
  458. });
  459. }
  460. }
  461. /* 初始化口上面的气体 */
  462. initSmokeMass() {
  463. if (!this.frontSmoke) this.frontSmoke = new Smoke('/model/img/texture-smoke.png', '#ffffff', 0, 0.4, 1.8, 100);
  464. if (!this.backSmoke) this.backSmoke = new Smoke('/model/img/texture-smoke.png', '#ffffff', 0, 0.4, 1.8, 100);
  465. }
  466. /* 口上面的气体 */
  467. setSmokePosition() {
  468. this.frontSmoke.setPoints();
  469. this.group?.add(this.frontSmoke.points);
  470. this.frontSmoke.points.position.set(-2.51, 2.51, 8.18);
  471. this.backSmoke.setPoints();
  472. this.group?.add(this.backSmoke.points);
  473. // this.backSmoke.points.position.set(47.07, -8.48, -1.83);
  474. }
  475. /** 初始化电机 */
  476. async initMotor() {
  477. // 前电机
  478. const motorGltf1 = await this.model.setModel('dj1');
  479. this.motorGroup1 = motorGltf1.scene as THREE.Group;
  480. this.motorGroup1?.position.set(0, 0, 0);
  481. this.motorGroup1.visible = false;
  482. this.motorGroup1.traverse((item) => {
  483. if (item.name === 'fan_blade003') {
  484. // @ts-ignore
  485. this.gearBack.gear1 = item as THREE.Group;
  486. } else if (item.name === 'fan_blade005') {
  487. // @ts-ignore
  488. this.gearBack.gear2 = item as THREE.Group;
  489. }
  490. });
  491. this.group?.add(this.motorGroup1);
  492. // 后电机
  493. const motorGltf2 = await this.model.setModel('dj2');
  494. this.motorGroup2 = motorGltf2.scene as THREE.Group;
  495. this.motorGroup2?.position.set(0, 0, 0);
  496. this.motorGroup2.visible = false;
  497. this.motorGroup2.traverse((item) => {
  498. if (item.name === 'fan_blade007') {
  499. // @ts-ignore
  500. this.gearFront.gear1 = item as THREE.Group;
  501. } else if (item.name === 'fan_blade006') {
  502. // @ts-ignore
  503. this.gearFront.gear2 = item as THREE.Group;
  504. }
  505. });
  506. this.group?.add(this.motorGroup2);
  507. }
  508. openOrCloseWindow(flag) {
  509. const _this = this;
  510. let endAngle = 0;
  511. if (flag == 'openWindow') {
  512. // 打开风窗
  513. endAngle = 1;
  514. } else {
  515. // 关闭风窗
  516. endAngle = 0;
  517. }
  518. gsap.to(this, {
  519. windowAngle: endAngle,
  520. duration: Math.abs(endAngle - this.windowAngle) * 10,
  521. ease: 'none',
  522. onUpdate: function () {
  523. _this.frontWindowGroup.children.forEach((mesh) => {
  524. mesh.rotation.z = _this.windowAngle;
  525. });
  526. },
  527. });
  528. }
  529. /** 初始化风窗 */
  530. initWindow() {
  531. if (!this.group) return;
  532. this.frontWindowGroup = new THREE.Group();
  533. this.frontWindowGroup.name = 'frontWindowGroup';
  534. this.backWindowGroup = new THREE.Group();
  535. this.backWindowGroup.name = 'backWindowGroup';
  536. if (this.group && this.group?.children.length > 0) {
  537. for (let i = this.group?.children.length - 1; i >= 0; i--) {
  538. const obj = this.group?.children[i];
  539. if (obj.type === 'Mesh' && obj.name && obj.name.startsWith('TC')) {
  540. const mesh = obj.clone();
  541. if (obj.name.startsWith('TC1')) {
  542. this.backWindowGroup.add(mesh);
  543. } else if (obj.name.startsWith('TC2')) {
  544. this.frontWindowGroup.add(mesh);
  545. }
  546. obj.removeFromParent();
  547. this.group?.remove(obj);
  548. }
  549. }
  550. }
  551. this.group?.add(this.backWindowGroup);
  552. this.group?.add(this.frontWindowGroup);
  553. }
  554. mountedThree() {
  555. return new Promise(async (resolve) => {
  556. this.model.setModel(this.modelName).then(async (gltf) => {
  557. this.group = gltf.scene;
  558. this.group?.position.set(-0.44, 29.88, 12.37);
  559. // this.initAnimation();
  560. // this.startAnimation()
  561. this.initSmokeMass();
  562. this.setSmokePosition();
  563. // this.startSmoke();
  564. // this.initSmoke();
  565. await this.initMotor();
  566. resolve(null);
  567. this.initWindow();
  568. this.openOrCloseWindow('openWindow');
  569. setTimeout(async () => {
  570. const videoPlayer1 = document.getElementById('main-player1')?.getElementsByClassName('vjs-tech')[0];
  571. if (videoPlayer1) {
  572. const mesh = renderVideo(this.group, videoPlayer1, 'player1');
  573. if (mesh) {
  574. mesh.scale.set(0.615, 0.57, 0.8);
  575. mesh.position.set(-43.783, 9.248, -21.78);
  576. this.group?.add(mesh);
  577. }
  578. videoPlayer1.addEventListener('fullscreen', (isFullscreen) => {
  579. if (isFullscreen) {
  580. this.model.orbitControls.enabled = false;
  581. } else {
  582. this.model.orbitControls.enabled = true;
  583. }
  584. });
  585. }
  586. }, 0);
  587. });
  588. });
  589. }
  590. destroy() {
  591. this.frontSmoke.clearSmoke();
  592. this.backSmoke.clearSmoke();
  593. this.model.clearGroup(this.motorGroup1);
  594. this.model.clearGroup(this.motorGroup2);
  595. this.model.clearGroup(this.group);
  596. this.gearFront.gear1?.geometry.dispose();
  597. this.gearFront.gear2?.geometry.dispose();
  598. this.gearFront.gear1?.material.dispose();
  599. this.gearFront.gear2?.material.dispose();
  600. this.gearBack.gear1?.geometry.dispose();
  601. this.gearBack.gear2?.geometry.dispose();
  602. this.gearBack.gear1?.material.dispose();
  603. this.gearBack.gear2?.material.dispose();
  604. this.model = null;
  605. this.group = null;
  606. }
  607. }
  608. export default mainWindRect;