index.html 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460
  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="utf-8">
  5. <link rel="icon" href="/favicon.ico" />
  6. <meta name="viewport"
  7. content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0, viewport-fit=cover">
  8. <meta http-equiv="Cache-control" content="no-cache">
  9. <meta http-equiv="Cache" content="no-cache">
  10. <meta name="apple-mobile-web-app-capable" content="yes" />
  11. <title>点名</title>
  12. <link rel="stylesheet" href="./css/index.css">
  13. <script src="js/three.min.js"></script>
  14. <script src="js/tween.min.js"></script>
  15. <!-- <script src="js/stats.min.js"></script> -->
  16. <script src="js/TrackballControls.js"></script>
  17. <script src="js/CSS3DRenderer.js"></script>
  18. <script src="js/vue.min.js"></script>
  19. <!-- 创建星星 -->
  20. <!-- <script src="js/star.js"></script> -->
  21. <!-- 数据位置 -->
  22. <script src="js/data.js"></script>
  23. </head>
  24. <body>
  25. <div id="container" style="pointer-events: none;"></div>
  26. <div id="vueBoxs">
  27. <div class="bss">
  28. <div class="pageTitle"></div>
  29. <div v-if="platform!=='modal'" class="iconBack" @click="onBack()"></div>
  30. <div id="menu" :class="[platform==='modal'?(imagePos==='right'?'menuRight':'menuLeft'):'']">
  31. <button id="table" :class="tableIndex <= 1 ? 'disabled' : ''" v-show="!animationStatus"
  32. @click="start()"></button>
  33. <button id="sphere" v-show="animationStatus" @click="closes()"></button>
  34. <button v-if="platform==='modal'" id="backBtn" @click="onBack()"></button>
  35. <!-- <button id="reset" style="margin-left:40px;" @click="resets()">照片墙</button> -->
  36. <!-- <button id="lists" @click="listShow = true">中奖名单</button> -->
  37. </div>
  38. </div>
  39. </div>
  40. </body>
  41. </html>
  42. <script>
  43. var table = getData()
  44. var tableLens = table.length;
  45. var camera, scene, renderer;
  46. var controls;
  47. var objects = [];
  48. var targets = { chaos: [], table: [], sphere: [], helix: [], grid: [] };
  49. var ang = 0, moving = false, objectsss, starBG, isMoving = false; //objectsss 为切换的图片的
  50. var cpx = 0, cpy = 0, cpz = 0, crx = 0, cry = 0, crz = 0;
  51. function init() {
  52. camera = new THREE.PerspectiveCamera(45, window.innerWidth / window.innerHeight, 1, 10000);
  53. camera.position.z = 1500;
  54. scene = new THREE.Scene();
  55. // table
  56. for (var i = 0; i < table.length; i++) {
  57. // 每个图标的盒子
  58. var element = document.createElement('div');
  59. element.className = 'element';
  60. element.style.backgroundColor = 'rgba(255,255,255)';
  61. element.style.borderRadius = '12px'
  62. if (i >= 32) {
  63. element.className = 'element hide'
  64. }
  65. // 图片
  66. var symbolBox = document.createElement('div');
  67. var symbol = document.createElement('img');
  68. symbolBox.className = 'symbolBox';
  69. symbol.className = 'symbol';
  70. symbol.src = table[i].img + '?imageMogr2/strip/format/png/size-limit/50k!';
  71. symbolBox.appendChild(symbol);
  72. element.appendChild(symbolBox);
  73. // 姓名
  74. var details = document.createElement('div');
  75. details.className = 'details';
  76. details.innerHTML = table[i].name;
  77. element.appendChild(details);
  78. // 图标变成3d内的对象,放入场景中
  79. var object = new THREE.CSS3DObject(element);
  80. object.position.x = Math.random() * 3400 - 1700;
  81. object.position.y = Math.random() * 3400 - 1700;
  82. object.position.z = Math.random() * 3400 - 1700;
  83. object.name = table[i].name;
  84. scene.add(object);
  85. objects.push(object);
  86. // 根据索引,定制位置
  87. var object = new THREE.Object3D();
  88. var ix = (i) % 8;
  89. object.position.y = 300;
  90. switch (table.length) {
  91. case 1:
  92. object.position.x = 0;
  93. break;
  94. case 2:
  95. object.position.x = (ix * 172) - (172 * 1 / 2);
  96. break;
  97. case 3:
  98. object.position.x = (ix * 172) - (172 * 2 / 2);
  99. break;
  100. case 4:
  101. object.position.x = (ix * 172) - (172 * 3 / 2);
  102. break;
  103. case 5:
  104. object.position.x = (ix * 172) - (172 * 4 / 2);
  105. break;
  106. case 6:
  107. object.position.x = (ix * 172) - (172 * 5 / 2);
  108. break;
  109. case 7:
  110. object.position.x = (ix * 172) - (172 * 6 / 2);
  111. break;
  112. default:
  113. var iy = Math.floor((i) / 8);
  114. object.position.x = (ix * 172) - 600;
  115. object.position.y = - (iy * 192) + 300;
  116. break;
  117. }
  118. targets.table.push(object);
  119. }
  120. // 创建切换图片
  121. var elements = document.createElement('div');
  122. elements.className = 'changeImgBoxs';
  123. // element.style.backgroundImage = "url(./img/pic.png)";
  124. elements.style.backgroundColor = 'rgba(255,255,255)';
  125. var symbolBox = document.createElement('div');
  126. var symbol = document.createElement('img');
  127. symbol.setAttribute("id", "changeImg");
  128. symbolBox.className = 'symbolBox2';
  129. symbol.className = 'symbol2';
  130. symbol.src = table[0].img + '?imageMogr2/strip/format/png/size-limit/50k!';
  131. symbolBox.appendChild(symbol);
  132. elements.appendChild(symbolBox);
  133. var details = document.createElement('div');
  134. details.setAttribute("id", "detailss");
  135. details.className = 'details';
  136. details.innerHTML = table[0].name;
  137. elements.appendChild(details);
  138. objectsss = new THREE.CSS3DObject(elements);
  139. objectsss.position.x = 0;
  140. objectsss.position.y = 20000;
  141. objectsss.position.z = 0;
  142. scene.add(objectsss);
  143. // 散乱随机位置
  144. var vector = new THREE.Vector3();
  145. for (var i = 0, l = objects.length; i < l; i++) {
  146. var phi = Math.acos(-1 + (2 * i) / l);
  147. var theta = Math.sqrt(l * Math.PI) * phi;
  148. var object = new THREE.Object3D();
  149. var py = Math.random() * 3400 - 1700;
  150. if (py < 400 && py > -400) { //防止停止时,图片位置不好挡住中央的图片
  151. if (py < 0) {
  152. py -= 400;
  153. } else {
  154. py += 400;
  155. }
  156. }
  157. object.position.x = Math.random() * 3400 - 1700;
  158. object.position.y = py;
  159. object.position.z = Math.random() * 3400 - 1700;
  160. object.lookAt(vector);
  161. targets.chaos.push(object);
  162. }
  163. // 球
  164. var vector = new THREE.Vector3();
  165. for (var i = 0, l = objects.length; i < l; i++) {
  166. var phi = Math.acos(-1 + (2 * i) / l);
  167. var theta = Math.sqrt(l * Math.PI) * phi;
  168. var object = new THREE.Object3D();
  169. object.position.x = 750 * Math.cos(theta) * Math.sin(phi);
  170. object.position.y = 750 * Math.sin(theta) * Math.sin(phi);
  171. object.position.z = 750 * Math.cos(phi);
  172. vector.copy(object.position).multiplyScalar(2);
  173. object.lookAt(vector);
  174. targets.sphere.push(object);
  175. }
  176. //渲染
  177. renderer = new THREE.CSS3DRenderer();
  178. renderer.setSize(window.innerWidth, window.innerHeight);
  179. renderer.domElement.style.position = 'absolute';
  180. document.getElementById('container').appendChild(renderer.domElement);
  181. //控制器
  182. controls = new THREE.TrackballControls(camera, renderer.domElement);
  183. controls.rotateSpeed = 0.5;
  184. controls.minDistance = 500;
  185. controls.maxDistance = 6000;
  186. controls.addEventListener('change', render);
  187. // 初始化
  188. transform(targets.table, 800);
  189. window.addEventListener('resize', onWindowResize, false);
  190. cpx = camera.position.x;
  191. cpy = camera.position.y;
  192. cpz = camera.position.z;
  193. crx = camera.rotation.x;
  194. cry = camera.rotation.y;
  195. crz = camera.rotation.z;
  196. }
  197. // 切换状态时动画
  198. function transform(targets, duration, type) {
  199. var scale = 1;
  200. if (type == undefined) {
  201. type = 0;
  202. }
  203. TWEEN.removeAll();
  204. for (var i = 0; i < objects.length; i++) {
  205. var object = objects[i];
  206. var target = targets[i];
  207. new TWEEN.Tween(object.position)
  208. .to({ x: target.position.x, y: target.position.y, z: target.position.z }, Math.random() * duration + duration)
  209. .easing(TWEEN.Easing.Exponential.InOut)
  210. .start();
  211. new TWEEN.Tween(object.rotation)
  212. .to({ x: target.rotation.x, y: target.rotation.y, z: target.rotation.z }, Math.random() * duration + duration)
  213. .easing(TWEEN.Easing.Exponential.InOut)
  214. .start();
  215. }
  216. new TWEEN.Tween(this)
  217. .to({}, duration * 2)
  218. .onUpdate(render)
  219. .start()
  220. // console.log(document.querySelectorAll('.element'), '1212')
  221. setTimeout(() => {
  222. console.log(moving, type)
  223. if (moving || type === 2) {
  224. const elements = document.querySelectorAll('.element')
  225. console.log(elements, 'eee')
  226. if (elements.length > 0) {
  227. elements.forEach((e, index) => {
  228. if (type === 2) {
  229. if (index >= 32) {
  230. e.classList.add('hide')
  231. }
  232. } else {
  233. e.classList.remove('hide')
  234. }
  235. })
  236. }
  237. }
  238. }, 400);
  239. }
  240. // 屏幕大小适配
  241. function onWindowResize() {
  242. camera.aspect = window.innerWidth / window.innerHeight;
  243. camera.updateProjectionMatrix();
  244. renderer.setSize(window.innerWidth, window.innerHeight);
  245. render();
  246. }
  247. function animate() {
  248. requestAnimationFrame(animate);
  249. TWEEN.update();
  250. // 旋转
  251. if (moving) {
  252. movings();
  253. }
  254. controls.update();
  255. // 星空位置与相机位置相对
  256. cpx = camera.position.x;
  257. cpy = camera.position.y;
  258. cpz = camera.position.z;
  259. crx = camera.rotation.x;
  260. cry = camera.rotation.y;
  261. crz = camera.rotation.z;
  262. }
  263. function render() {
  264. renderer.render(scene, camera);
  265. }
  266. var numsss = 0, srcss = '', txtsss = '';
  267. function movings() {
  268. // 相机旋转
  269. ang += Math.PI / 50;
  270. camera.position.x = Math.cos(ang) * 2000;
  271. camera.position.z = Math.sin(ang) * 2000;
  272. camera.position.y = 0;
  273. // 相机方向重置
  274. camera.up.x = 0;
  275. camera.up.y = 1;
  276. camera.up.z = 0;
  277. // 图片方向固定
  278. objectsss.rotation.y = -ang + Math.PI / 2;
  279. //中间图片切换
  280. numsss = Math.floor(Math.random() * tableLens);
  281. if (numsss == tableLens) {
  282. numsss = tableLens - 1;
  283. }
  284. srcss = table[numsss].img + '?imageMogr2/strip/format/png/size-limit/50k!';
  285. txtsss = table[numsss].name;
  286. changeImg.src = srcss;
  287. detailss.innerHTML = txtsss;
  288. }
  289. // 停止时,图片放大动画
  290. function objDeal(obj, nums) {
  291. var option = {
  292. x: obj.scale.x,
  293. y: obj.scale.y,
  294. z: obj.scale.z,
  295. };
  296. var tween = new TWEEN.Tween(option).to({
  297. x: nums,
  298. y: nums,
  299. z: nums,
  300. }, 300).delay(100).onUpdate(function () {
  301. obj.scale.x = this.x;
  302. obj.scale.y = this.y;
  303. obj.scale.z = this.z;
  304. isMoving = true;
  305. }).onComplete(function () {
  306. isMoving = false;
  307. }).start();
  308. }
  309. // 初始化数据
  310. function renderData(data) {
  311. table = data
  312. tableLens = data.length
  313. vm.tableIndex = tableLens
  314. init();
  315. animate();
  316. }
  317. function preloadImages(imageUrls, callback) {
  318. let images = {};
  319. let loadedImages = 0;
  320. let numImages = imageUrls.length;
  321. // 创建一个Image对象,并设置src属性来开始加载图片
  322. imageUrls.forEach((url) => {
  323. images[url] = new Image();
  324. images[url].onload = function () {
  325. // 当图片加载完成时,更新计数器
  326. loadedImages++;
  327. if (loadedImages === numImages && callback) {
  328. callback(images); // 所有图片都加载完毕,调用回调函数
  329. }
  330. };
  331. images[url].onerror = function () {
  332. console.error('Could not load image at ' + url);
  333. loadedImages++;
  334. };
  335. images[url].src = url;
  336. });
  337. }
  338. //点击事件及部分判断
  339. var vm = new Vue({
  340. el: '#vueBoxs',
  341. data: {
  342. animationStatus: false,
  343. tableIndex: 0,
  344. listShow: false,
  345. platform: "",
  346. imagePos: ""
  347. },
  348. methods: {
  349. keyDowns: function () {
  350. this.closes();
  351. },
  352. //开始
  353. start: function () {
  354. if (this.tableIndex <= 1) {
  355. return
  356. }
  357. moving = true;
  358. this.animationStatus = true;
  359. objectsss.position.y = 0;
  360. transform(targets.sphere, 1000);
  361. objDeal(objectsss, 1);
  362. // music.play();
  363. // this.ckPrice();
  364. },
  365. //结束
  366. closes: function () {
  367. // music.pause();
  368. if (!moving) {
  369. return;
  370. }
  371. // music2.play();
  372. moving = false;
  373. this.animationStatus = false;
  374. // this.choosePerson();
  375. transform(targets.chaos, 250, 1);
  376. objDeal(objectsss, 1.8);
  377. },
  378. //回到照片墙的状态
  379. resets: function () {
  380. moving = false;
  381. camera.position.z = 1500;
  382. camera.position.x = 0;
  383. camera.position.y = 0;
  384. camera.up.x = 0;//相机以哪个方向为上方
  385. camera.up.y = 1;
  386. camera.up.z = 0;
  387. //图片重置方向
  388. objectsss.rotation.y = 0;
  389. objectsss.position.y = 20000;
  390. objectsss.scale.set(1, 1, 1);
  391. // 星空背景图重置
  392. transform(targets.table, 400, 2);
  393. },
  394. onBack() {
  395. // 返回时先结束
  396. this.closes()
  397. this.resets()
  398. window.parent.postMessage({
  399. api: 'callBack',
  400. loading: false,
  401. }, '*');
  402. },
  403. handleUrlParmas(){
  404. const params=getUrlParams(location.href)
  405. params?.platform&&(this.platform = params?.platform)
  406. params?.imagePos&&(this.imagePos = params?.imagePos)
  407. }
  408. },
  409. created: function () {
  410. // init();
  411. // animate();
  412. // const imageUrls = table.map(value => value.img + '?imageMogr2/strip/format/png/size-limit/50k!')
  413. // console.log(imageUrls.length, 'length')
  414. // console.log(imageUrls, 'imageUrls')
  415. // preloadImages(imageUrls, function (images) {
  416. // // console.log('All images loaded.');
  417. // // // 这里可以访问images对象,它是一个包含预加载图片的对象,
  418. // // // 其中键是图片的URL,值是Image对象。
  419. // init();
  420. // animate();
  421. // setTimeout(() => {
  422. // animate();
  423. // }, 100);
  424. // });
  425. // getStar();
  426. // this.ckPrice();
  427. this.handleUrlParmas()
  428. }
  429. })
  430. function getUrlParams(url) {
  431. let urlStr = url.split('?')[1]
  432. const urlSearchParams = new URLSearchParams(urlStr)
  433. const result = Object.fromEntries(urlSearchParams.entries())
  434. return result
  435. }
  436. </script>