Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Ease on spinning wheel

I'm creating a Wheel of Fortune like game and currently working on the wheel. I want to add ease out effect to the wheel to make the spin realistic. I have no prior knowledge on easing, so I implemented the code in 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>

Ease out works fine but it's not complete. Spin doesn't end. I want to end it when it's too slow. I don't know what I should be checking to end the spinning. How do I end it?


Update: Alex's answer solved the problem, but I wanted to add another feature. Spin always ends at 100. This isn't realistic. It should vary. So I used a dynamic max velocity instead of 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);
}
like image 712
akinuri Avatar asked Feb 26 '16 12:02

akinuri


2 Answers

Try this=) Updated:

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>
like image 50
Alexandr Kudryashov Avatar answered Sep 21 '22 21:09

Alexandr Kudryashov


A better ease

Though the answer given solves the problem there is a general solution to the ease function. The reason I give this answer even though there is an accepted answer is that the accepted answer suffers from one major drawback. It requires that the animation runs at a constant frame rate. If the frame rate drops the wheel will appear to slow down. If you include the time in the equation the use of the derivative to change the position will change the stop position making the whole function indeterminate (angularVelocity += .0035 * timeStep) does not work, there is no linear solution to this method. (I have worked in the (gambling) gaming industry and an indeterminate function would never pass accreditation, nor would an operator want to use such a function in their machines).

You need a function that you can precisely set the prize before the spin, you want an equation that is not affected by frame time, and not on the previous machine state.

The solution is a general ease function of which there are many flavours. It is simply the anti derivative of the function in the previous answer in a more usable form.

Ease

If you graph the math function y = x2 you will see a parabola centered at x = 0 and at x = 1 it has risen to y = 1. If you consider the values from x = 0 to x = 1 to be our range this simple formula has given us an easeOut function. For any value of x in the range x >= 0and x <= 1 we get an adjusted value for y. Effectively converting a linear value into a curve.

But need EaseIn

We want the opposite, we need the formula that has the parabola upside down, with the maxima at y = 1 when x = 1 and at x = 0 we want y = 0. This is easy to do with the above formula.

  • First to flip it make the result negative. y = -x2
  • Then we need to move the maxima up the y axis by 1 so add one: y = 1-x2
  • Now we need to move the parabola to the right by one unit so sub one from x. y = 1 - (x - 1)2

So now we have a new curve for x >= 0 and x <= 1 that eases in as x approaches 1.

Let's create two functions for the easeIn and easeOut, we will put them in a ease object to keep it organised. In the functions we want to ensure that if x < 0 and x > 1 we don't get bad values so we need to clamp x to the range 0 to 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);
    }
}

Normalised Units.

But 0 to 1 does not fit my solution. No is does not but one is easy to change via multiplication.

So you have the wheel you want to spin. You know how long you want it to spin and you know where you want it to stop.

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

As we want to reset the wheel many times lets create a function to get a spin

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;
}

An event handler for the spin button

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));
    }
}

So now we are ready to animate. I will assume that the animation is running continuously.

// 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)

This method will always stop the wheel at the correct time and correct position. The ease function gives a value from 0 to 1 and we use that to get wheel position with a simple multiplication. The input value is simply the normalised time. Which just needs us to divide the time by the length of time for the spin. As the ease functions are clamped we don't have to worry about values out side the range as the math takes care of that for us.

Changing the curve.

The ease function I provided does not give you the ability to change the ease amount. We may want to have the ease very fast at the start and then ease in very slowly, or have it only have a very slight ease. Luck has it that in math the power of a number between 0 and 1 never gets greater than 1. We can use this to change the ease amount by providing a value to set the ease power. It requires a little modification of the easeIn function.

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));
    }
}

You now have a general purpose ease function. Changing the value of pow will change the ease. As raising a number to the power of a fraction is the same as the root of the inverse of the fraction 40.5 = 22 the ease function also does ease out when the value is less than 1. If pow is one then there is no ease. A value greater then one starts to ease the function in.

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

While I am here I might as well add the function for 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))
    }
}

There are many ease functions and they can do a wide range of motions, they can even return outside the range to give bounces at the start and end if wanted, or even add notches (step downs) to the spin. The important thing is that the input value of the ease function is with in the normalised range and that the start x = 0 and end x = 1 end at y = 0 and y = 1 respectively. This makes the function totally determinate and time invariant, You can go backward as easily as forward.

like image 45
Blindman67 Avatar answered Sep 17 '22 21:09

Blindman67