目的:通过js实现小球碰撞并实现动量守恒

canvas我们就不多说了,有用着呢。

我们可以通过canvas画2D图形(圆、方块、三角形等等)3D图形(球体、正方体等待)。

当然这只是基础的皮毛而已,canvas的强大之处在于可以做游戏,导入模型,粒子效果,实现漫游又或者全景和VR。

这里我们介绍纯js写的2D小球碰撞。(主要是博主的Three.js不咋地)

好吧,老规矩,先上图!

额。。。很尴尬的是博主的截图功底不咋地,没有截下碰撞的瞬间。

话不多说,开始教程。

首先我们需要创建画布给它一个id方便后面监听处理。

<body>
<canvas id="myCanvas"></canvas>
</body>

然后是Js代码

//声明画布大小为屏幕的1/3
var width = window.innerWidth/3;
var height = window.innerHeight/3;
var canvas = document.getElementById("myCanvas");
canvas.width = width;
canvas.height = height;
//创建2d画笔
var ctx = canvas.getContext("2d");
//填充颜色设置为黑色(背景色)
ctx.fillStyle = "#000";
//将整个画布填充
ctx.fillRect(0,0,width,height);

这是Canvas最基本的操作,我们解释一下fillRect(x,y,width,height);这个函数。

x和y填充的起始坐标点,width和height是填充区域的宽和高。

由于我们创建的是带有物理性质的小球,所以我们用一个函数封装创建小球的代码。

以后创建小球直接调用它就行了。

function Ball(x,y,vx,vy,ax,ay,size,rou,color,ctx){
//参数传值
//x,y为坐标点 vx,vy为小球水平和垂直方向上的速度 ax,ay为加速度
//size 为大小 rou为密度 color颜色 ctx画笔
this.x = x;
this.y = y;
this.vx = vx;
this.vy = vy;
this.rou = rou;
this.size = size;
this.ax = ax;
this.ay = ay;
this.m = Math.PI*this.size*this.size*rou;//求出质量 this.draw = function(ctx){
ctx.fillStyle=color;
//console.log(this.x, this.y,this.size);
ctx.beginPath();
ctx.arc(this.x, this.y, this.size, 0, Math.PI*2,false);//画圆
ctx.fill();
ctx.closePath();
}
this.draw(ctx);
}

接下来实例化出两个小球。

//碰撞检测 动量守恒
//x,y,vx,vy,ax,ay,size,rou,color,ctx
var balla = new Ball(20,0.5*height,5,-3,0,0,8,1,"#ff0",ctx);
var ballb = new Ball(width-20,0.5*height,-3,5,0,0,13,1,"#0ff",ctx);
var ballc = new Ball(width/2,0.5*height,7,4,0,0,13,1,"#0ff",ctx);

然后我们封装了一个函数来实现小球是实时更新。

function animation(){

        //小球的速度等于速度加上加速度
balla.vx+= balla.ax;
balla.vy+=balla.ay;
//小球的位移等于小球现在的坐标加上速度
balla.x+= balla.vx;
balla.y+=balla.vy; ballb.vx+= ballb.ax;
ballb.vy+=ballb.ay;
ballb.x+= ballb.vx;
ballb.y+=ballb.vy;
//基于距离的碰撞检测
var pointdis=(balla.x-ballb.x)*(balla.x-ballb.x)+(balla.y-ballb.y)*(balla.y-ballb.y);//坐标距离
var pointsize=(balla.size+ballb.size)*(balla.size+ballb.size);//半径距离
if( pointdis <= pointsize)
{
console.log("haha");
//这里是能量守恒公式
var ballavx =((balla.m-ballb.m)*balla.vx+2*ballb.m*ballb.vx)/(balla.m+ballb.m);
var ballavy =((balla.m-ballb.m)*balla.vy+2*ballb.m*ballb.vy)/(balla.m+ballb.m);
var ballbvx=((ballb.m-balla.m)*ballb.vx+2*balla.m*balla.vx)/(balla.m+ballb.m);
var ballbvy=((ballb.m-balla.m)*ballb.vy+2*balla.m*balla.vy)/(balla.m+ballb.m);
balla.vx = ballavx;
balla.vy = ballavy; ballb.vx = ballbvx;
ballb.vy = ballbvy;
//小Bug改进
if(Math.abs(balla.vx-ballb.vx)<0.01&&Math.abs(balla.vy-ballb.vy)<0.01)
{
console.log(balla.vx);
balla.vx=-balla.vx;
balla.vy=-balla.vy;
return;
}
} //判断是否碰撞到画布的边缘
if(balla.x+balla.size>=width||balla.x-balla.size<=0)
{
balla.vx*=-0.98;
}
if(balla.y+balla.size>=height||balla.y-balla.size<=0)
{
balla.vy*=-0.98;
} if(ballb.x+ballb.size>=width||ballb.x-ballb.size<=0)
{
ballb.vx*=-0.98;
}
if(ballb.y+ballb.size>=height||ballb.y-ballb.size<=0)
{
ballb.vy*=-0.98;
} //清空画布,画出小球
ctx.fillStyle = "#000";
ctx.fillRect(0,0,width,height);
balla.draw(ctx);
ballb.draw(ctx);
//console.log(ballb.vy); }

最后我们让他30毫秒更新一次。

    setInterval(animation,30);

OK,又大功告成了。自己动手试试吧!

懒人福利!!!

完整代码。

<html>
<head>
<style>
body{margin:0;}
</style>
</head>
<body>
<canvas id="myCanvas"></canvas>
</body>
<script>
//声明画布大小为屏幕的1/3
var width = window.innerWidth/3;
var height = window.innerHeight/3;
var canvas = document.getElementById("myCanvas");
canvas.width = width;
canvas.height = height;
//创建2d画笔
var ctx = canvas.getContext("2d");
//填充颜色设置为黑色(背景色)
ctx.fillStyle = "#000";
//将整个画布填充
ctx.fillRect(0,0,width,height); //碰撞检测 动量守恒
//x,y,vx,vy,ax,ay,size,rou,color,ctx
var balla = new Ball(20,0.5*height,5,-3,0,0,8,1,"#ff0",ctx);
var ballb = new Ball(width-20,0.5*height,-3,5,0,0,13,1,"#0ff",ctx);
var ballc = new Ball(width/2,0.5*height,7,4,0,0,13,1,"#0ff",ctx); setInterval(animation,30);
function animation(){ //小球的速度等于速度加上加速度
balla.vx+= balla.ax;
balla.vy+=balla.ay;
//小球的位移等于小球现在的坐标加上速度
balla.x+= balla.vx;
balla.y+=balla.vy; ballb.vx+= ballb.ax;
ballb.vy+=ballb.ay;
ballb.x+= ballb.vx;
ballb.y+=ballb.vy;
//基于距离的碰撞检测
var pointdis=(balla.x-ballb.x)*(balla.x-ballb.x)+(balla.y-ballb.y)*(balla.y-ballb.y);//坐标距离
var pointsize=(balla.size+ballb.size)*(balla.size+ballb.size);//半径距离
if( pointdis <= pointsize)
{
console.log("haha");
//这里是能量守恒公式
var ballavx =((balla.m-ballb.m)*balla.vx+2*ballb.m*ballb.vx)/(balla.m+ballb.m);
var ballavy =((balla.m-ballb.m)*balla.vy+2*ballb.m*ballb.vy)/(balla.m+ballb.m);
var ballbvx=((ballb.m-balla.m)*ballb.vx+2*balla.m*balla.vx)/(balla.m+ballb.m);
var ballbvy=((ballb.m-balla.m)*ballb.vy+2*balla.m*balla.vy)/(balla.m+ballb.m);
balla.vx = ballavx;
balla.vy = ballavy; ballb.vx = ballbvx;
ballb.vy = ballbvy;
//小Bug改进
if(Math.abs(balla.vx-ballb.vx)<0.01&&Math.abs(balla.vy-ballb.vy)<0.01)
{
console.log(balla.vx);
balla.vx=-balla.vx;
balla.vy=-balla.vy;
return;
}
} //判断是否碰撞到画布的边缘
if(balla.x+balla.size>=width||balla.x-balla.size<=0)
{
balla.vx*=-0.98;
}
if(balla.y+balla.size>=height||balla.y-balla.size<=0)
{
balla.vy*=-0.98;
} if(ballb.x+ballb.size>=width||ballb.x-ballb.size<=0)
{
ballb.vx*=-0.98;
}
if(ballb.y+ballb.size>=height||ballb.y-ballb.size<=0)
{
ballb.vy*=-0.98;
} //清空画布,画出小球
ctx.fillStyle = "#000";
ctx.fillRect(0,0,width,height);
balla.draw(ctx);
ballb.draw(ctx);
//console.log(ballb.vy); }
function Ball(x,y,vx,vy,ax,ay,size,rou,color,ctx){
//参数传值
//x,y为坐标点 vx,vy为小球水平和垂直方向上的速度 ax,ay为加速度
//size 为大小 rou为密度 color颜色 ctx画笔
this.x = x;
this.y = y;
this.vx = vx;
this.vy = vy;
this.rou = rou;
this.size = size;
this.ax = ax;
this.ay = ay;
this.m = Math.PI*this.size*this.size*rou;//求出质量 this.draw = function(ctx){
ctx.fillStyle=color;
//console.log(this.x, this.y,this.size);
ctx.beginPath();
ctx.arc(this.x, this.y, this.size, 0, Math.PI*2,false);//画圆
ctx.fill();
ctx.closePath();
}
this.draw(ctx);
}
</script>
</html>

欢迎交流学习!!!

不定时随缘更新。

Canvas+Js制作动量守恒的小球碰撞的更多相关文章

  1. js实现多个小球碰撞

    实现思路:小球的移动,是通过改变小球的left和top值来改变,坐标分别为(x,y)当x/y值加到最大,即加到父级的宽度或者高度时,使x值或者y值减小,同理当x值或者y值减到最小时,同样的使x值或者y ...

  2. HTML5 Canvas彩色小球碰撞运动特效

    脚本简介 HTML5 Canvas彩色小球碰撞运动特效是一款基于canvas加面向对象制作的运动小球动画特效.   效果展示 http://hovertree.com/texiao/html5/39/ ...

  3. 利用canvas制作乱跑的小球

    canvas制作乱跑的小球 说明:将下面的代码放到html的body就可以,键盘控制上(W)下(S)左(A)右(D) <body> <canvas id="canvas&q ...

  4. 小游戏——js+h5[canvas]+cs3制作【五子棋】小游戏

    五子棋小游戏学习—— 有一个问题是,棋盘线的颜色,在canvas中,明明设置了灰色,但在我的预览中还是黑色的,很重的颜色. 以下是复刻的源码: <!DOCTYPE html> <ht ...

  5. js实现小球碰撞游戏

    效果图:  效果图消失只是截的gif图的问题 源码: <!DOCTYPE html> <html lang="en"> <head> <m ...

  6. 使用canvas通过js制作一个小型英雄抓怪兽的2D小游戏

    首先,这是一个HTML: <!DOCTYPE html> <html lang="en"> <head> <meta charset=&q ...

  7. 使用WebGL + Three.js制作动画场景

    使用WebGL + Three.js制作动画场景 3D图像,技术,打造产品,还有互联网:这些只是我爱好的一小部分. 现在,感谢WebGL的出现-一个新的JavaScriptAPI,它可以在不依赖任何插 ...

  8. Expression Design与Blend制作滚动的小球动画教程

    原文:Expression Design与Blend制作滚动的小球动画教程 一,开发工具 Microsoft Expression Design & Blend 4.0 (3.0亦可). 这两 ...

  9. js制作带有遮罩弹出层实现登录小窗口

    要实现的效果如下 点击“登录”按钮后,弹出登录小窗口,并且有遮罩层(这个名词还是百度知道的,以前只知道效果,却不知道名字) 在没有点击“登录”按钮之前登录小窗口不显示,点击“登录”按钮后小窗口显示,并 ...

随机推荐

  1. Zookeeper 系列(三)Zookeeper API

    Zookeeper 系列(三)Zookeeper API 本节首先介绍 Zookeeper 的 Shell 命令,再对 Java 操作 Zookeeper 的三种方式进行讲解,本节先介绍 Zookee ...

  2. 递归生成treeview树形节点(没有用递归函数之后会有补充,这里只用系统的内置方法去生成)

    using System;using System.Collections.Generic;using System.ComponentModel;using System.IO;using Syst ...

  3. AJAX学习必备三本书

    <AJAX基础教程>AJAX必备图书之一.国内发行的第一本AJAX图书,也是目前最好的AJAX入门书,如果您是AJAX新手,此书是最好的入门图书.本书基本包括了实现Ajax需要了解的大部分 ...

  4. 2018.10.22 bzoj1742: Grazing on the Run 边跑边吃草(区间dp)

    传送门 区间dp入门题. 可以想到当前吃掉的草一定是一个区间(因为经过的草一定会吃掉). 然后最后一定会停在左端点或者右端点. f[i][j][0/1]f[i][j][0/1]f[i][j][0/1] ...

  5. hdu-1116(欧拉回路+并查集)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1116 思路:将字符串的头元素和尾元素视为图的x,y节点,然后合并x,y. 如果这个图不连通,则门不能打 ...

  6. Django入门指南-第10章:Django Admin 介绍(完结)

    在浏览器中打开该URL:http://127.0.0.1:8000/admin/ 我们可以检查一切是否正常,打开URL http://127.0.0.1:8000 我们首先创建一个管理员帐户: pyt ...

  7. arduino 驱动电调

    #include <TimerOne.h> #define PPMPIN 7 ; //0-9 ; void setup() { // put your setup code here, t ...

  8. ansible-playbook 主机变量2

    ansible-playbook 配置 hosts 后可以指定变量,通过-k 可以交互输入密码,也可以将密码写在 hosts 文件中. 入口 yaml 文件中通过 {{ ** }} 获取变量,命令行通 ...

  9. day3之函数的初始及进阶

    函数初始 函数的定义与调用 ''' def 函数名 (参数): 函数体 函数名:设定与变量相同 执行函数: 函数名() ''' 函数的返回值 # 函数返回值 return ''' 1.遇到return ...

  10. 201709015工作日记--上下文的理解,ASM

    1.Android上下文理解 Android上下文对象,在Context中封装一个所谓的“语境”,Activity.Service.Application都继承自Context,所以在这三者创建时都会 ...