轻松纺车

时间:2016-02-26 12:39:45

标签: javascript animation canvas

我创造了一个像游戏一样的财富之轮,目前正在开车。我想为车轮添加缓和效果,使旋转变得逼真。我之前没有关于缓和的知识,因此我在Introduction to Easing in JavaScript中实现了代码。

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>
&#13;
&#13;
&#13;

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

更新:Alex的回答解决了这个问题,但我想添加其他功能。旋转总是以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);
}

2 个答案:

答案 0 :(得分:5)

试试这个=) 的更新

&#13;
&#13;
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;
}
&#13;
<div id="container">
	<img src="" id="arrow" />
	<canvas id="canvas" width="410" height="410"></canvas>
	<input type="button" value="Spin" onclick="startSpin()"/>
</div>
&#13;
class PostForm(forms.ModelForm):
    class Meta:
        model = Post
        fields = '__all__'
    captcha = NoReCaptchaField()
&#13;
&#13;
&#13;

答案 1 :(得分:1)

更轻松

虽然给出的答案解决了问题,但是缓解功能有一个通用的解决方案。即使有一个公认的答案,我给出这个答案的原因是接受的答案有一个主要缺点。它要求动画以恒定的帧速率运行。如果帧速率下降,则车轮将显示为减速。如果在方程中包含时间,使用导数来改变位置将改变停止位置,使整个函数不确定(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 的反函数的根相同时,easy函数也会在值小于1.如果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.

Bonus easeInOut

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

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时结束。这使得函数完全确定并且时间不变,您可以像前进一样轻松地返回。