Внезапно обратился к лотерейному софту, написанному в предыдущей компании (для годового собрания компании). Я чувствую себя очень эмоционально.В ТМ 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 вывод
Код написан несколько лет назад, очень фигня, производительность не считал, да и не делал. Просто идея, которую я сделал в то время, была более ясной. Всем предоставьте несколько актуальных бизнес-идей.
демонстрационный адрес(Бесплатный облачный сервер Tencent, срок действия истек 26 февраля 2022 г.)