繁体   English   中英

轻松纺车

[英]Ease on spinning wheel

我正在创造一个像游戏一样的财富之轮,目前正在开车。 我想为车轮添加缓和效果,使旋转变得逼真。 我之前没有关于缓动的知识,所以我在JavaScript中使用了Easing简介中的代码。

的jsfiddle

 var canvas = document.getElementById("canvas"); var context = canvas.getContext("2d"); var img = new Image(); img.src = ""; context.resetTransform(); context.translate(205, 205); context.drawImage(img, -201,-201,402,402); var angle = 0; var angularVelocity = .01; function draw() { // ease out angularVelocity += .0035; angle -= (1 / angularVelocity); context.resetTransform(); context.clearRect(0, 0, canvas.width, canvas.height); context.translate(205, 205); context.rotate(angle * Math.PI/180); context.drawImage(img, -201,-201,402,402); spin = requestAnimationFrame(draw); } function startSpin() { spin = requestAnimationFrame(draw); } 
 body { background: gainsboro; margin: 0; } #container { position: relative; margin: 20px; margin-left: 40px; display: table; } canvas { background: white; box-shadow: 0 0 3px rgba(0,0,0,.2); } input { display: block; padding: 5px 10px; margin: auto; } img { height: 40px; position: absolute; top: 185px; left: -33px; } 
 <div id="container"> <img src="http://i1115.photobucket.com/albums/k544/akinuri/arrow.png" /> <canvas id="canvas" width="410" height="410"></canvas> <input type="button" value="Spin" onclick="startSpin()"/> </div> 

轻松工作正常,但不完整。 旋转不会结束。 当它太慢时,我想结束它。 我不知道应该检查什么来结束纺纱。 我该如何结束?


更新 :亚历克斯的答案解决了这个问题,但我想添加另一个功能。 旋转总是以100结束。这是不现实的。 它应该有所不同。 所以我使用了动态最大速度而不是1

的jsfiddle

var angle           = 0;
var angularVelocity = 0.001;
var minVelocity     = 0.9;
var maxVelocity     = 0;

function draw() {
    angularVelocity += .0025;
    angle -= (1 / angularVelocity);

    context.resetTransform();
    context.clearRect(0, 0, canvas.width, canvas.height);
    context.translate(205, 205);
    context.rotate(angle * Math.PI/180);
    context.drawImage(img, -201,-201,402,402);

    if (angularVelocity < maxVelocity){ 
        spin = requestAnimationFrame(draw);
    }
}

function startSpin() {
    angularVelocity = 0.001;
    maxVelocity     = (Math.random() * 1.5).toFixed(2);
    while (maxVelocity < minVelocity) {
        maxVelocity     = (Math.random() * 1.5).toFixed(2);
    }
    spin = requestAnimationFrame(draw);
}

试试这个=) 更新:

 var canvas = document.getElementById("canvas"); var context = canvas.getContext("2d"); var img = new Image(); img.src = ""; context.resetTransform(); context.translate(205, 205); context.drawImage(img, -201,-201,402,402); var angle = 0; var angularVelocity = .01; function draw() { // ease out angularVelocity += .0035; angle -= (1 / angularVelocity); context.resetTransform(); context.clearRect(0, 0, canvas.width, canvas.height); context.translate(205, 205); context.rotate(angle * Math.PI/180); context.drawImage(img, -201,-201,402,402); if (angularVelocity+0.1<1){ spin = requestAnimationFrame(draw); } } function startSpin() { angularVelocity = .01; spin = requestAnimationFrame(draw); } 
 body { background: gainsboro; margin: 0; } #container { position: relative; margin: 20px; margin-left: 40px; display: table; } canvas { background: white; box-shadow: 0 0 3px rgba(0,0,0,.2); } input { display: block; padding: 5px 10px; margin: auto; } img { height: 40px; position: absolute; top: 185px; left: -33px; } 
 <div id="container"> <img src="" id="arrow" /> <canvas id="canvas" width="410" height="410"></canvas> <input type="button" value="Spin" onclick="startSpin()"/> </div> 

更好的方便

虽然给出的答案解决了问题,但是缓解功能有一个通用的解决方案。 即使有一个公认的答案,我给出这个答案的原因是接受的答案有一个主要缺点。 它要求动画以恒定的帧速率运行。 如果帧速率下降,则车轮将显示为减速。 如果在方程中包含时间,使用导数来改变位置将改变停止位置,使整个函数不确定(angularVelocity + = .0035 * timeStep)不起作用,这种方法没有线性解。 (我曾在(赌博)游戏行业工作,不确定的职能永远不会通过认证,运营商也不希望在他们的机器中使用这样的功能)。

您需要一个可以在旋转之前精确设置奖励的功能,您需要一个不受帧时间影响的方程式,而不是之前的机器状态。

该解决方案是一种易于使用的功能,其中有许多口味。 它只是前一个答案中函数的反导数,是一种更实用的形式。

缓解

如果您绘制数学函数y = x 2,您将看到一个以x = 0为中心的抛物线,并且在x = 1时它已经上升到y = 1.如果您认为从x = 0到x = 1的值是我们的范围这个简单的公式给了我们一个easeOut函数。 对于x >= 0x <= 1范围内的任何x值,我们得到y的调整值。 有效地将线性值转换为曲线。

但需要EaseIn

我们想要相反的,我们需要抛物线上下颠倒的公式,当x = 1时,y = 1时的最大值和x = 0时我们想要y = 0.这很容易用上面的公式来做。

  • 首先翻转它会使结果为负。 y = -x 2
  • 然后我们需要将y轴上的最大值移动1,因此添加一个:y = 1-x 2
  • 现在我们需要将抛物线向右移动一个单位,因此从x开始。 y = 1 - (x - 1) 2

所以现在我们有一个新的曲线, x >= 0x <= 1 ,当x接近1时,它会缓和。

让我们为easeIn和easeOut创建两个函数,我们将它们放在一个easy对象中以保持其有序。 在函数中我们要确保如果x < 0x > 1我们不会得到错误的值,所以我们需要将x钳位到0到1的范围。

var easing = {
    easeOut : function( x ) {        
        return Math.pow( Math.min(1, Math.max(0, x) ), 2);
    },
    easeIn : function( x ) {
        return 1 - Math.pow( Math.min(1, Math.max(0, x) ) - 1, 2);
    }
}

标准化单位。

但0到1不适合我的解决方案。 不是没有,但通过乘法很容易改变。

所以你有想要旋转的轮子。 你知道你想要它旋转多久,你知道你希望它停在哪里。

const NUMBER_PRIZES = 10; // number of prizes on the wheel.
const PRIZE_CENTER = Math.PI / NUMBER_PRIZES; // center of the prize
const DEG360 = Math.PI * 2; // use PI * two a few times so make a constant to represent 360 deg.
var timeToSpin = 10;  // ten seconds.
var whereToStop = 0; // Start pos
var startTime = 0 ; // if the value of start time is undefined this will mean we want to start
var startPos = 0; // wheel start pos

因为我们想要多次重置轮子,所以我们可以创建一个旋转功能

var spinTheWheel = function(toPrize){
    whereToStop %= DEG360;  // Normalise the current position. This will not move it
    startPos = whereToStop;  // we need the start pos
    // We need to move the wheel to the start so we can get the correct prize
    whereToStop += DEG360 - whereToStop; 

    whereToStop += DEG360 * 2;  // At least two turns
    // Now get a random prize add a random position to the wheel stopping pos
    whereToStop += (toPrize / NUMBER_PRIZES) * DEG360;
    // we want the wheel to stop in the center of the prize so rotate by half a prize sector
    whereToStop += PRIZE_CENTER;
}

旋转按钮的事件处理程序

var spin = function () {
    var time = new Date().valueOf();
    // make sure we are not spinning
    if(time - startTime > timeToSpin * 1000){
        startTime = undefined;  // reset start time
        // get a random prize
        spinTheWheel ( Math.floor(Math.Random() * NUMBER_PRIZES));
    }
}

所以现在我们准备动画了。 我将假设动画持续运行。

// function to draw the wheel.
function drawWheel (wheelPos) {} // todo

// main animation loop
function update(time) {  // request frame gives us the time as an argument.

    var spinTime, wheel;  // vars we need

    // is there a new spin ???
    if(startTime === undefined) { // new start 
        startTime = time;   // set the time now;
    }

    // get the amount of time the wheel has been spinning
    spinTime = (time - startTime);

    // now as the spin time is in seconds and we want normalise it to a range of 0 - 1;
    // we dont care if the spin time goes out of range as the easeOut function is clamped
    spinTime /= timeToSpin * 1000; // so divide by number of milliseconds.

    // Now use the easeOut function to get the wheel pos in terms of spin time
    wheel = easing.ease(spinTime);

    // the wheel pos is in the range 0 to 1 so scale it to fit the requiered spin
    wheel *= whereToStop - startPos;

    // Now add the startPos back to the wheel pos.
    wheel += startPos;

    // Draw the wheel.
    drawWheel(wheel);

    // to know if the wheel is at a prize
    if(spinTime >= 1) { 
        // todo 
        // your prize code here
    }


    requestAnimationFrame(update)
}
// start it.
requestAnimationFrame(update)

此方法将始终在正确的时间和正确的位置停止车轮。 缓动函数给出一个从0到1的值,我们使用它来通过简单的乘法获得车轮位置。 输入值只是标准化时间。 这只需要我们将时间除以旋转的时间长度。 由于缓和功能被限制,我们不必担心数值超出范围,因为数学会为我们处理。

改变曲线。

我提供的缓解功能并不能让您更改缓动量。 我们可能想要在开始时非常快速地放松,然后非常缓慢地放松,或者只是非常轻松。 幸运的是,在数学中,0到1之间的数字的功率永远不会大于1.我们可以通过提供设置缓和功率的值来使用它来改变缓动量。 它需要稍微修改easeIn功能。

var easing = {
    easeOut : function( x ) {        
        return Math.pow( Math.min(1, Math.max(0, x) ), 2);
    },
    easeIn : function( x ) {
        return 1 - Math.pow( Math.min(1, Math.max(0, x) ) - 1, 2);
    },    
    ease : function( x , pow) {  // add the power. the ease amount
        // need to clamp the power so it does not go in the negative        
        return 1 - Math.pow(1 - Math.min(1, Math.max(0, x) ), Math.max(0, pow));
    }
}

您现在拥有通用缓动功能。 改变战俘的价值将改变轻松。 将一个数字提高到一个分数的幂是与分数4的倒数的根相同0.5 = 2 2当该值小于1时,easy函数也会缓和。如果pow是1那么就没有缓解。 大于1的值开始缓解函数。

easing.ease(val, 0.5); // is an ease out
easing.ease(val, 2); // is an ease in
easing.ease(val, 1); // is no easing at all
easing.ease(val, 6); // is a very strong easein. fast at start slow to stop.

奖金easeInOut

当我在这里时,我不妨添加easeInOut的功能

var easing = {
    easeOut : function( x ) {        
        return Math.pow( Math.min(1, Math.max(0, x) ), 2);
    },
    easeIn : function( x ) {
        return 1 - Math.pow( Math.min(1, Math.max(0, x) ) - 1, 2);
    },    
    ease : function( x , pow) {  // add the power. the ease amount
        // need to clamp the power so it does not go in the negative        
        return 1 - Math.pow(1 - Math.min(1, Math.max(0, x) ), Math.max(pow));
    },
    easeInOut = function (x, pow) {
        pow = Math.max(0, pow); // clamp pow to >= 0
        x = Math.min(1, Math.max(0, x)); // clamp x >= 0 and <= 1
        var xx = Math.pow(x, pow);
        return xx / (xx + Math.pow(1-x, pow))
    }
}

有许多缓和功能,它们可以进行各种运动,甚至可以返回到范围之外,如果需要,可以在开始和结束时弹跳,甚至可以为旋转添加凹口(降级)。 重要的是,缓动函数的输入值在归一化范围内,并且起始x = 0和结束x = 1分别在y = 0和y = 1时结束。 这使得函数完全确定并且时间不变,您可以像前进一样轻松地向后移动。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM