fanLocalDual.threejs.base.ts 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381
  1. import * as THREE from 'three';
  2. // import { setModalCenter } from '/@/utils/threejs/util';
  3. import Smoke from '../../comment/threejs/Smoke';
  4. import { CSS3DObject, CSS3DSprite } from 'three/examples/jsm/renderers/CSS3DRenderer';
  5. import * as dat from 'dat.gui';
  6. // const gui = new dat.GUI();
  7. // gui.domElement.style = 'position:absolute;top:100px;left:10px;z-index:99999999999999';
  8. class ModelContext {
  9. model;
  10. // modelName = 'jbfj-hd';
  11. modelName = 'jbfj-dual';
  12. /** 本模型的根3D对象 */
  13. group?: THREE.Object3D;
  14. /** 本模型所包含的所有元素合集 */
  15. private elements: unknown[] = [];
  16. /** 本模型支持的 Object3DGroup 模块 */
  17. private modules: {
  18. /** 模块名称 */
  19. name: string;
  20. /** 控制该模块所用的上下文 */
  21. context: THREE.Object3D;
  22. /** 控制时行为声明 */
  23. behavior: (context: THREE.Object3D) => void;
  24. }[] = [];
  25. constructor(model) {
  26. this.model = model;
  27. }
  28. addLight() {
  29. // optional implementation
  30. }
  31. /** 设置模型类型并切换,不同的类型通常对应不同的具体模型,在模型总控制器下的具体模型会根据传入的参数彼此交互、切换 */
  32. setModelType(modelType: string, data: any) {
  33. this.modules.forEach(({ name, context, behavior }) => {
  34. if (name === modelType) {
  35. behavior(context);
  36. }
  37. });
  38. }
  39. mountedThree() {
  40. return new Promise((resolve) => {
  41. this.model.setGLTFModel([this.modelName]).then(async (gltf) => {
  42. this.group = gltf[0];
  43. if (this.group) {
  44. // this.group.scale.set(2, 2, 2);
  45. // setModalCenter(this.group);
  46. this.addLight();
  47. this.setModelPosition();
  48. this.initModules().then(resolve);
  49. }
  50. });
  51. });
  52. }
  53. destroy() {
  54. if (!this.model) return;
  55. this.elements.forEach((element) => {
  56. this.model.clearGroup(element);
  57. });
  58. }
  59. // 设置模型位置
  60. setModelPosition() {
  61. if (!this.group) return;
  62. this.group.scale.set(0.6, 0.6, 0.6);
  63. // const ff = gui.addFolder(`位置调整`);
  64. // ff.add(this.group.position, 'x', -100, 100);
  65. // ff.add(this.group.position, 'y', -100, 100);
  66. // ff.add(this.group.position, 'z', -100, 100);
  67. this.group.position.set(0, 0, -60);
  68. this.group.rotation.y = Math.PI / 2;
  69. }
  70. // hideElements(eles: THREE.Object3D[]) {
  71. // eles.forEach((g) => {
  72. // g.visible = false;
  73. // });
  74. // }
  75. // showElements(eles: THREE.Object3D[]) {
  76. // eles.forEach((g) => {
  77. // g.visible = true;
  78. // });
  79. // }
  80. weakElements(eles: unknown[]) {
  81. eles.forEach((g) => {
  82. if (g instanceof Smoke) {
  83. g.oldOpacityFactor = 0.4;
  84. }
  85. if (g instanceof CSS3DObject) {
  86. g.element.style.setProperty('opacity', '0.5');
  87. }
  88. });
  89. }
  90. strongElements(eles: unknown[]) {
  91. eles.forEach((g) => {
  92. if (g instanceof Smoke) {
  93. g.oldOpacityFactor = 0.75;
  94. }
  95. if (g instanceof CSS3DObject) {
  96. g.element.style.setProperty('opacity', '1');
  97. }
  98. });
  99. }
  100. startAnimation(eles: unknown[]) {
  101. eles.forEach((g) => {
  102. if (g instanceof Smoke) {
  103. g.startSmoke();
  104. }
  105. });
  106. }
  107. stopAnimation(eles: unknown[]) {
  108. const smokes = eles.filter((g) => {
  109. return g instanceof Smoke;
  110. });
  111. return Promise.all(smokes.map((e) => e.stopSmoke()));
  112. }
  113. /** 核心方法,初始化本模型的各个模块,这些模块可以实现特定场景的展示、控制等功能 */
  114. async initModules() {
  115. if (this.elements.length > 0) return;
  116. // 右侧风机-主风机进风
  117. const curveFan1Right = this.generateSmokePath(['dian', 'dian1', 'dian2', 'dian3', 'dian4', 'dian5', 'dian6', 'dian7', 'dian8'], true);
  118. // 右侧风机-备风机进风
  119. const curveFan2Right = this.generateSmokePath(['dian9', 'dian10', 'dian2', 'dian3', 'dian4', 'dian5', 'dian6', 'dian7', 'dian8'], true);
  120. // 左侧风机-主风机进风
  121. const curveFan1Left = this.generateSmokePath(['dian11', 'dian12', 'dian15', 'dian16', 'dian17'], true);
  122. // 左侧风机-备风机进风
  123. const curveFan2Left = this.generateSmokePath(['dian14', 'dian13', 'dian15', 'dian16', 'dian17'], true);
  124. // 右侧巷道-回风前段
  125. const curveTunnelRight = this.generateSmokePath(['dian18', 'dian19', 'dian20']);
  126. // 左侧巷道-回风前段
  127. const curveTunnelLeft = this.generateSmokePath(['dian20', 'dian22']);
  128. // 左侧巷道-回风全长
  129. const curveTunnelMajor = this.generateSmokePath(['dian21', 'dian22']);
  130. const group1 = new THREE.Group();
  131. const smokeFan1Right = new Smoke('/model/img/texture-smoke.png', '#ffffff', 10, 0.75, 0.5, 400);
  132. smokeFan1Right.setPath(curveFan1Right);
  133. this.elements.push(smokeFan1Right);
  134. const smokeFan2Right = new Smoke('/model/img/texture-smoke.png', '#ffffff', 10, 0.75, 0.5, 400);
  135. smokeFan2Right.setPath(curveFan2Right);
  136. this.elements.push(smokeFan2Right);
  137. const smokeFan1Left = new Smoke('/model/img/texture-smoke.png', '#ffffff', 10, 0.75, 0.5, 400);
  138. smokeFan1Left.setPath(curveFan1Left);
  139. this.elements.push(smokeFan1Left);
  140. const smokeFan2Left = new Smoke('/model/img/texture-smoke.png', '#ffffff', 10, 0.75, 0.5, 400);
  141. smokeFan2Left.setPath(curveFan2Left);
  142. this.elements.push(smokeFan2Left);
  143. const smokeTunnelRight = new Smoke('/model/img/texture-smoke.png', '#ffffff', 0, 0.35, 1.5, 200);
  144. smokeTunnelRight.setPath(curveTunnelRight);
  145. this.elements.push(smokeTunnelRight);
  146. const smokeTunnelLeft = new Smoke('/model/img/texture-smoke.png', '#ffffff', 0, 0.35, 1.5, 200);
  147. smokeTunnelLeft.setPath(curveTunnelLeft);
  148. this.elements.push(smokeTunnelLeft);
  149. const smokeTunnelMajor = new Smoke('/model/img/texture-smoke.png', '#ffffff', 0, 0.35, 1.5, 200);
  150. smokeTunnelMajor.setPath(curveTunnelMajor);
  151. this.elements.push(smokeTunnelMajor);
  152. await smokeFan1Right.setPoints();
  153. this.group?.add(smokeFan1Right.points);
  154. await smokeFan2Right.setPoints();
  155. this.group?.add(smokeFan2Right.points);
  156. await smokeFan1Left.setPoints();
  157. this.group?.add(smokeFan1Left.points);
  158. await smokeFan2Left.setPoints();
  159. this.group?.add(smokeFan2Left.points);
  160. await smokeTunnelRight.setPoints();
  161. this.group?.add(smokeTunnelRight.points);
  162. await smokeTunnelLeft.setPoints();
  163. this.group?.add(smokeTunnelLeft.points);
  164. await smokeTunnelMajor.setPoints();
  165. this.group?.add(smokeTunnelMajor.points);
  166. const fanRightSelectors = [
  167. {
  168. query: '#inputBox2',
  169. position: [-70, 0, -15],
  170. scale: 0.1,
  171. },
  172. {
  173. query: '#T1_1',
  174. position: [100, 15, -42],
  175. scale: 0.2,
  176. },
  177. {
  178. query: '#T1_2',
  179. position: [30, 15, -37],
  180. scale: 0.175,
  181. },
  182. ];
  183. const fanLeftSelectors = [
  184. {
  185. query: '#inputBox3',
  186. position: [-70, 0, 36],
  187. scale: 0.1,
  188. },
  189. {
  190. query: '#T2_1',
  191. position: [110, 15, -80],
  192. scale: 0.2,
  193. },
  194. {
  195. query: '#T2_2',
  196. position: [47, 15, -74],
  197. scale: 0.175,
  198. },
  199. ];
  200. const commonSelectors = [
  201. {
  202. query: '#T3',
  203. position: [-40, 15, -65],
  204. scale: 0.15,
  205. },
  206. ];
  207. const fanLeftSprites = this.initCssElement(fanLeftSelectors);
  208. const fanRightSprites = this.initCssElement(fanRightSelectors);
  209. const commonSprites = this.initCssElement(commonSelectors);
  210. // 双巷道的通风机都开启有4种情况
  211. this.modules.push({
  212. name: 'fan1RightOpen&fan1LeftOpen',
  213. context: group1,
  214. behavior: () => {
  215. // this.weakElements(this.elements);
  216. this.stopAnimation(this.elements).then(() => {
  217. // this.strongElements();
  218. this.startAnimation([smokeFan1Right, smokeFan1Left, smokeTunnelRight, smokeTunnelMajor]);
  219. });
  220. },
  221. });
  222. this.modules.push({
  223. name: 'fan2RightOpen&fan1LeftOpen',
  224. context: group1,
  225. behavior: () => {
  226. this.stopAnimation(this.elements).then(() => {
  227. this.startAnimation([smokeFan2Right, smokeFan1Left, smokeTunnelRight, smokeTunnelMajor]);
  228. });
  229. },
  230. });
  231. this.modules.push({
  232. name: 'fan1RightOpen&fan2LeftOpen',
  233. context: group1,
  234. behavior: () => {
  235. this.stopAnimation(this.elements).then(() => {
  236. this.startAnimation([smokeFan1Right, smokeFan2Left, smokeTunnelRight, smokeTunnelMajor]);
  237. });
  238. },
  239. });
  240. this.modules.push({
  241. name: 'fan2RightOpen&fan2LeftOpen',
  242. context: group1,
  243. behavior: () => {
  244. this.stopAnimation(this.elements).then(() => {
  245. this.startAnimation([smokeFan2Right, smokeFan2Left, smokeTunnelRight, smokeTunnelMajor]);
  246. });
  247. },
  248. });
  249. // 只有一个风机启动有4种情况
  250. this.modules.push({
  251. name: 'fan1RightOpen',
  252. context: group1,
  253. behavior: () => {
  254. this.stopAnimation(this.elements).then(() => {
  255. this.startAnimation([smokeFan1Right, smokeTunnelMajor]);
  256. });
  257. },
  258. });
  259. this.modules.push({
  260. name: 'fan2RightOpen',
  261. context: group1,
  262. behavior: () => {
  263. this.stopAnimation(this.elements).then(() => {
  264. this.startAnimation([smokeFan2Right, smokeTunnelMajor]);
  265. });
  266. },
  267. });
  268. this.modules.push({
  269. name: 'fan1LeftOpen',
  270. context: group1,
  271. behavior: () => {
  272. this.stopAnimation(this.elements).then(() => {
  273. this.startAnimation([smokeFan1Left, smokeTunnelRight, smokeTunnelLeft]);
  274. });
  275. },
  276. });
  277. this.modules.push({
  278. name: 'fan2LeftOpen',
  279. context: group1,
  280. behavior: () => {
  281. this.stopAnimation(this.elements).then(() => {
  282. this.startAnimation([smokeFan2Left, smokeTunnelRight, smokeTunnelLeft]);
  283. });
  284. },
  285. });
  286. // 只有一个风机启动有2种告示牌情况
  287. this.modules.push({
  288. name: 'fanLeftStrong',
  289. context: group1,
  290. behavior: () => {
  291. this.weakElements(this.elements);
  292. this.strongElements([...fanLeftSprites, ...commonSprites]);
  293. },
  294. });
  295. this.modules.push({
  296. name: 'fanRightStrong',
  297. context: group1,
  298. behavior: () => {
  299. this.weakElements(this.elements);
  300. this.strongElements([...fanRightSprites, ...commonSprites]);
  301. },
  302. });
  303. }
  304. /** 初始化css元素,将css元素选择器传入,该方法会将这些元素按顺序放入传入的锚点中 */
  305. initCssElement(selectors: { query: string; position: number[]; scale: number }[]) {
  306. const arr: CSS3DSprite[] = [];
  307. selectors.forEach(({ query, position, scale }) => {
  308. const element = document.querySelector(query) as HTMLElement;
  309. if (element) {
  310. const css3D = new CSS3DSprite(element);
  311. this.elements.push(css3D);
  312. arr.push(css3D);
  313. css3D.name = query;
  314. css3D.scale.set(scale, scale, scale);
  315. // const ff = gui.addFolder(`css元素${query}`);
  316. // ff.add(css3D.position, 'x', -100, 100);
  317. // ff.add(css3D.position, 'y', -100, 100);
  318. // ff.add(css3D.position, 'z', -100, 100);
  319. css3D.position.set(position[0], position[1], position[2]);
  320. this.group?.add(css3D);
  321. }
  322. });
  323. return arr;
  324. }
  325. /** 生成适用于 Smoke 的曲线数据,输入途径点,输出路径,如果是进风类型,首个线段将有扩散效果,出风则是末尾线段有扩散效果 */
  326. generateSmokePath(namelist: string[], airIn?: boolean, airOut?: boolean) {
  327. if (!this.group) return;
  328. const result: any[] = [];
  329. const points: THREE.Vector3[] = namelist.map((name) => {
  330. const obj = this.group?.getObjectByName(name);
  331. return obj?.position as THREE.Vector3;
  332. });
  333. for (let index = 1; index < points.length; index++) {
  334. const path0 = points[index - 1];
  335. const path1 = points[index];
  336. const path = {
  337. path0,
  338. path1,
  339. isSpread: false,
  340. spreadDirection: 0,
  341. };
  342. if (airIn) {
  343. // 首个线段需要扩散,由大变小
  344. path.isSpread = index === 1;
  345. path.spreadDirection = -1;
  346. }
  347. if (airOut) {
  348. // 首个线段需要扩散,由小变大
  349. path.isSpread = index === points.length - 1;
  350. path.spreadDirection = 1;
  351. }
  352. if (!airIn && !airOut) {
  353. path.isSpread = false;
  354. path.spreadDirection = 1;
  355. }
  356. result.push(path);
  357. }
  358. return result;
  359. }
  360. }
  361. export default ModelContext;