3D страница лотереи (threejs)

внешний интерфейс three.js
3D страница лотереи (threejs)

Внезапно обратился к лотерейному софту, написанному в предыдущей компании (для годового собрания компании). Я чувствую себя очень эмоционально.В ТМ 30+ человек, а нарисовано около 15. Код до сих пор пишется мной, но я не могу рисовать. (Правда, персонаж поднялся, а я с ума сошла)

Несколько слов эмоций:

1. Интерфейс случайный, он действительно псевдослучайный! ! ! (За себя узнал при тестировании рандома)

2. Пожалуйста, называйте меня шовным монстром.

1. Эффект

демонстрационный адрес(Бесплатный облачный сервер Tencent, срок действия истек 26 февраля 2022 г.)

Схема эффекта выглядит следующим образом:

2. Основной эффект

1. «Периодическая таблица элементов»

Фотостена? лотерея? В период до написания мне довелось зайти на блог и увидеть трехмерную версию «Периодической таблицы элементов», написанную другим великим богом, эффект в целом аналогичный. (ссылку забыл на тот момент, сейчас дамтри демонстрационные ссылки)

Два самых важных там.

Затем измените элементы и подсказки на фотографии и названия для лотереи, а для рендеринга используйте CSS3DObject;

// table
  for ( var i = 0; i < table.length; i += 2 ) {
    // 每个图标的盒子
    var element = document.createElement( 'div' );
    element.className = 'element';
    element.style.backgroundColor = 'rgba(0,127,127,' + ( Math.random() * 0.5 + 0.25 ) + ')';
    // 索引
    var number = document.createElement( 'div' );
    number.className = 'number';
    number.textContent = (i/2) + 1;
    element.appendChild( number );
    // 图片
    var symbolBox = document.createElement( 'div' );
    var symbol = document.createElement( 'img' );
    symbolBox.className = 'symbolBox';
    symbol.className = 'symbol';
    symbol.src = table[i];
    symbolBox.appendChild(symbol);
    element.appendChild( symbolBox );
    // 姓名
    var details = document.createElement( 'div' );
    details.className = 'details';
    details.innerHTML = table[ i + 1 ];
    element.appendChild( details );
    // 图标变成3d内的对象,放入场景中
    var object = new THREE.CSS3DObject( element );
    object.position.x = Math.random() * 3400 - 1700;
    object.position.y = Math.random() * 3400 - 1700;
    object.position.z = Math.random() * 3400 - 1700;
    object.name=table[ i + 1 ];
    scene.add( object );
    objects.push( object );
    // 根据索引,定制位置
    var object = new THREE.Object3D();
    var iy = Math.floor((i/2)/9);
    var ix = (i/2)%9;
    object.position.x = (ix * 140 ) -540;
    object.position.y = - ( iy * 180 ) + 480;
    targets.table.push( object );
  }

2. Звездная базовая карта, звездная анимация

Просто периодическая таблица элементов слишком проста. Фоном должно быть звездное небо или что-то в этом роде, желательно Млечный Путь!

Обои со звездным небом высокой четкости слишком велики. Если это GIF с небольшим количеством движения, он будет больше. А на низком пикселе эффекта нет вообще. (адаптировано к экрану 1920 г.)

Итак, я использовал низкопиксельное изображение + анимацию звезды на холсте, чтобы улучшить фон!

Анимация звездного холста. (тоже нашел в сети, ссылку забыл) (спасибо предыдущему великому богу +1)

var canvas = document.getElementById('canvas'), 
  ctx = canvas.getContext('2d'), 
  w = canvas.width = window.innerWidth, 
  h = canvas.height = window.innerHeight, 
 
  hue = 217, 
  stars = [], 
  count = 0, 
  maxStars = 1300;//星星数量 
 
var canvas2 = document.createElement('canvas'), 
  ctx2 = canvas2.getContext('2d'); 
canvas2.width = 100; 
canvas2.height = 100; 
var half = canvas2.width / 2, 
  gradient2 = ctx2.createRadialGradient(half, half, 0, half, half, half); 
gradient2.addColorStop(0.025, '#CCC'); 
gradient2.addColorStop(0.1, 'hsl(' + hue + ', 61%, 33%)'); 
gradient2.addColorStop(0.25, 'hsl(' + hue + ', 64%, 6%)'); 
gradient2.addColorStop(1, 'transparent'); 
 
ctx2.fillStyle = gradient2; 
ctx2.beginPath(); 
ctx2.arc(half, half, half, 0, Math.PI * 2); 
ctx2.fill(); 
 
// End cache 
 
function random(min, max) { 
  if (arguments.length < 2) { 
    max = min; 
    min = 0; 
  } 
 
  if (min > max) { 
    var hold = max; 
    max = min; 
    min = hold; 
  } 
 
  return Math.floor(Math.random() * (max - min + 1)) + min; 
} 
 
function maxOrbit(x, y) { 
  var max = Math.max(x, y), 
    diameter = Math.round(Math.sqrt(max * max + max * max)); 
  return diameter / 2; 
  //星星移动范围,值越大范围越小, 
} 
 
var Star = function() { 
 
  this.orbitRadius = random(maxOrbit(w, h)); 
  this.radius = random(60, this.orbitRadius) / 8;  
  //星星大小 
  this.orbitX = w / 2; 
  this.orbitY = h / 2; 
  this.timePassed = random(0, maxStars); 
  this.speed = random(this.orbitRadius) / 50000;  
  //星星移动速度 
  this.alpha = random(2, 10) / 10; 
 
  count++; 
  stars[count] = this; 
} 
 
Star.prototype.draw = function() { 
  var x = Math.sin(this.timePassed) * this.orbitRadius + this.orbitX, 
    y = Math.cos(this.timePassed) * this.orbitRadius + this.orbitY, 
    twinkle = random(10); 
 
  if (twinkle === 1 && this.alpha > 0) { 
    this.alpha -= 0.05; 
  } else if (twinkle === 2 && this.alpha < 1) { 
    this.alpha += 0.05; 
  } 
 
  ctx.globalAlpha = this.alpha; 
  ctx.drawImage(canvas2, x - this.radius / 2, y - this.radius / 2, this.radius, this.radius); 
  this.timePassed += this.speed; 
} 
 
for (var i = 0; i < maxStars; i++) { 
  new Star(); 
} 
 
function animation() { 
  ctx.globalCompositeOperation = 'source-over'; 
  ctx.globalAlpha = 0.5; //尾巴 
  ctx.fillStyle = 'hsla(' + hue + ', 64%, 6%, 2)'; 
  ctx.fillRect(0, 0, w, h) 
 
  ctx.globalCompositeOperation = 'lighter'; 
  for (var i = 1, l = stars.length; i < l; i++) { 
    stars[i].draw(); 
  }; 
 
  window.requestAnimationFrame(animation); 
} 
 
animation();

3. Фотографии лотереи

Фотография лотереи должна быть в центре фотошара. (также в 3D)

Таким же образом используйте CSS3DObject для рендеринга

// 创建切换图片
var elements = document.createElement( 'div' );
elements.className = 'changeImgBoxs';
// element.style.backgroundImage = "url(./img/pic.png)";
elements.style.backgroundColor = 'rgba(0,127,127,' + ( Math.random() * 0.5 + 0.25 ) + ')';

var symbolBox = document.createElement( 'div' );
var symbol = document.createElement( 'img' );
symbol.setAttribute("id", "changeImg");
symbolBox.className = 'symbolBox2';
symbol.className = 'symbol2';
symbol.src = table[0];
symbolBox.appendChild(symbol);
elements.appendChild( symbolBox );

var details = document.createElement( 'div' );
details.setAttribute("id", "detailss");
details.className = 'details';
details.innerHTML = table[1];
elements.appendChild( details );

objectsss = new THREE.CSS3DObject( elements );
objectsss.position.x = 0;
objectsss.position.y = 20000;
objectsss.position.z = 0;
scene.add( objectsss );

2. Анимация

Все основы готовы, только анимация.

1. Анимация Tween для фотостены, фотошара и разбросанного состояния фото.

Нажмите, чтобы нарисовать: Фотостена ---> Фотошар

Нажмите, чтобы остановить: Photo Ball ---> Photo Scatter (Похоже, Photo Ball взорвался)

В threejs для анимации движения обычно используется tween.js.

Сначала сохраните координатные точки трех состояний фотостены, шара и рассеяния, а затем переключите состояние лотереи, переключитесь одновременно и переключите положение фотографии через анимацию, тем самым реализуя анимацию.

// 表格坐标 (在初始化)
object.position.x = Math.random() * 3400 - 1700;
    object.position.y = Math.random() * 3400 - 1700;
    object.position.z = Math.random() * 3400 - 1700;
    object.name=table[ i + 1 ];
    scene.add( object );
    objects.push( object );
    // 根据索引,定制位置
    var object = new THREE.Object3D();
    var iy = Math.floor((i/2)/9);
    var ix = (i/2)%9;
    object.position.x = (ix * 140 ) -540;
    object.position.y = - ( iy * 180 ) + 480;
    targets.table.push( object );

// 球的 照片坐标
var vector = new THREE.Vector3();
for ( var i = 0, l = objects.length; i < l; i ++ ) {
	var phi = Math.acos( -1 + ( 2 * i ) / l );
	var theta = Math.sqrt( l * Math.PI ) * phi;
	var object = new THREE.Object3D();
	object.position.x = 750 * Math.cos( theta ) * Math.sin( phi );
	object.position.y = 750 * Math.sin( theta ) * Math.sin( phi );
	object.position.z = 750 * Math.cos( phi );
	vector.copy( object.position ).multiplyScalar( 2 );
	object.lookAt( vector );
	targets.sphere.push( object );
}
// 散乱随机位置
var vector = new THREE.Vector3();
for ( var i = 0, l = objects.length; i < l; i ++ ) {
	var phi = Math.acos( -1 + ( 2 * i ) / l );
	var theta = Math.sqrt( l * Math.PI ) * phi;
	var object = new THREE.Object3D();
	var py = Math.random() * 3400 - 1700;
	if(py<400&&py>-400){    //防止停止时,图片位置不好挡住中央的图片中间空出点位置
		if(py<0){
			py-=400;
		}else{
			py+=400;
		}
	}
	object.position.x = Math.random() * 3400 - 1700;
	object.position.y = py;					
	object.position.z = Math.random() * 3400 - 1700;
	object.lookAt( vector );
	targets.chaos.push( object );
}	

tweenjs патч анимация

// 切换状态时动画
function transform( targets, duration ,type) {
  var scale = 1;
  if(type==undefined){
    type=0;
  }
  TWEEN.removeAll();
  for ( var i = 0; i < objects.length; i ++ ) {
    var object = objects[ i ];
    var target = targets[ i ];
    new TWEEN.Tween( object.position )
      .to( { x: target.position.x, y: target.position.y, z: target.position.z }, Math.random() * duration + duration )
      .easing( TWEEN.Easing.Exponential.InOut )
      .start();
    new TWEEN.Tween( object.rotation )
      .to( { x: target.rotation.x, y: target.rotation.y, z: target.rotation.z }, Math.random() * duration + duration )
      .easing( TWEEN.Easing.Exponential.InOut )
      .start();                                 
  }
  new TWEEN.Tween( this )
    .to( {}, duration * 2 )
    .onUpdate( render )
    .start();
}

2. Анимация во время лотереи

На данный момент в основном две анимации: переключение фото, вращение фотошара

Переключение фото, среднее изображение src можно переключать случайным образом

Фотошарик крутится? Зачем заморачиваться, не проще ли камеру повернуть?

Поэтому используйте: контроллер трека TrackballControls.js. Просто поверните камеру

var numsss =0,srcss='',txtsss='';function movings(){
  // 相机旋转
  ang += Math.PI/50;
  camera.position.x = Math.cos(ang)*2000;
  camera.position.z = Math.sin(ang)*2000; 
  camera.position.y = 0;
  // 相机方向重置
  camera.up.x = 0;
  camera.up.y = 1;
  camera.up.z = 0;
  // 图片方向固定
  objectsss.rotation.y =-ang+Math.PI/2;
  //中间图片切换
  numsss = Math.floor(Math.random()*tableLens);
  if(numsss==tableLens){
    numsss = tableLens-1;
  }
  srcss = table[numsss*2];
  txtsss =  table[numsss*2+1];
  changeImg.src = srcss;
  detailss.innerHTML = txtsss;
}

3. Выигрышная картинка будет увеличена после розыгрыша лотереи.

Лотерея окончена, остановите вращение, и выигрышная картинка увеличится

// 停止时,图片放大动画function objDeal(obj,nums){
  var option = {
    x: obj.scale.x,  
    y: obj.scale.y, 
    z: obj.scale.z,                 
  };
  var tween = new TWEEN.Tween(option).to({
    x:nums,
    y:nums,
    z:nums,
  },300).delay(100).onUpdate(function() {
    obj.scale.x = this.x;
    obj.scale.y = this.y;
    obj.scale.z = this.z;
    isMoving = true;
  }).onComplete(function(){
    isMoving = false;                
  }).start();
}

3. Улучшить детализацию и добавить музыку

Для лотереи светового эффекта недостаточно, нужна музыка. (Музыку в то время предоставила сестра HR)

На данный момент добавлено 2 музыки: После нажатия на розыгрыш захватывающая музыка "Pig Rush". Музыка при выигрыше в лотерею, рингтон Bling Bling.

<!--抽奖音乐标签-->
<audio id="music" src="./music/04-抽奖音乐 猪突猛進.mp3" loop></audio>
<!--抽奖音乐标签-->
<audio id="music2" src="./music/9629.mp3"></audio>

//开始
start:function(){
  if(vm.spic.img!=''){
    alert("抽完咯~~~完咯~~~咯~~~,在中奖名单中清空,再来一次?");
    return;
  }
  moving = true;              
  objectsss.position.y = 0;
  transform( targets.sphere, 1000 );
  objDeal(objectsss,1);
  music.play(); // 开始音乐
  this.ckPrice();           
},
//结束
closes:function(){
  music.pause(); // 关闭音乐
  if(!moving){
    return;
  } 
  music2.play(); // 播放中奖音乐
  moving = false;
  this.choosePerson();
  transform( targets.chaos, 250 ,1);
  objDeal(objectsss,1.8);   
},

IV вывод

Код написан несколько лет назад, очень фигня, производительность не считал, да и не делал. Просто идея, которую я сделал в то время, была более ясной. Всем предоставьте несколько актуальных бизнес-идей.

код,GitHub.com/whitedog/some…

демонстрационный адрес(Бесплатный облачный сервер Tencent, срок действия истек 26 февраля 2022 г.)