从零开始手把手教你使用javascript+canvas开发一个塔防游戏06塔发射子弹

简介: 从零开始手把手教你使用javascript+canvas开发一个塔防游戏06塔发射子弹

项目演示



image.png


项目演示地址:

体验一下

项目源码:

项目源码

代码结构

image.png

本节做完效果

image.png


新增bullet.js

//子弹类
 function Bullet(cxt,img,type,enemy,level,x,y,radius){
    this.cxt = cxt;
    this.img = img;
    this.x = x;
    this.y = y;
    this.radius = radius;
    this.radiusAll = radius * 2;
    //子弹类型
    this.type = type;
    this.enemy = enemy;
    //子弹级别
    this.level = level;
    this.sp = 5;
    //穿刺弹的移动速度
    this.lineAngle = this.getLineAngle();
    if(x > enemy.x+20)this.lineAngle.xsp *= -1;
    if(y > enemy.y+20)this.lineAngle.ysp *= -1;
    this.lineEnemy = {};
}
Bullet.prototype = {
    //子弹在图像中的对应
    bulletMap : [{x:0,y:0},{x:10,y:0},{x:20,y:0},{x:30,y:0},{x:40,y:0}],
    //画子弹
    draw : function(){
        Canvas.drawImg(this.cxt,this.img,this.bulletMap[this.type].x,this.bulletMap[this.type].y,this.radiusAll,this.radiusAll,this.x,this.y,this.radiusAll,this.radiusAll);
    },
    //更新子弹信息
    update : function(enemyList){
        var bulletInfo = BulletType[this.type]["level_"+this.level],
            enemy;
        //判断是否是穿刺子弹
        if(this.type == 3){
            this.x += this.lineAngle.xsp;
            this.y += this.lineAngle.ysp;
            if(this.y >= 500 || this.y <= 0 || this.x >= 500 || this.x <= 0){
                return this.over();
            }
            //遍历每个敌人
            for(var i=0,l=enemyList.length;i<l;i++){
                enemy = enemyList[i];
                if(!enemy)continue;
                //如果穿刺过,跳过
                if(this.lineEnemy[enemy.num])continue;
                //判断子弹击中敌人
                if(T.circleInCircle(this,{x:enemy.x+20,y:enemy.y+20,radius:20})){
                    this.lineHurt(enemy);
                    //将穿刺过的敌人保存
                    this.lineEnemy[enemy.num] = true;
                }
            }
        }
        //非穿刺弹
        else{
            //敌人死了,子弹结束
            if(!this.enemy.islive)return this.over();
            //移动的速度
            var sp = this.getLineAngle();
            if(this.x > this.enemy.x+20)this.x -= sp.xsp;
            else if(this.x < this.enemy.x +20)this.x += sp.xsp;
            if(this.y > this.enemy.y+20)this.y -= sp.ysp;
            else if(this.y < this.enemy.y + 20)this.y += sp.ysp;
            if(this.y >= 500 || this.y <= 0 || this.x >= 500 || this.x <= 0){
                return this.over();
            }
            //判断子弹击中敌人
            if(T.circleInCircle(this,{x:this.enemy.x+20,y:this.enemy.y+20,radius:20})){
                return this.over(this.enemy);
            }
        }
    },
    //获取子弹对于敌人的移动速度
    getLineAngle : function(){
        var ydif = Math.abs(this.y-this.enemy.y-15),
            xdif = Math.abs(this.x-this.enemy.x-15),
            xsp,ysp;
        if(ydif >= xdif){
            ysp = this.sp;
            xsp = Math.floor(this.sp * (xdif / ydif));
        }
        else {
            xsp = this.sp;
            ysp = Math.floor(this.sp * (ydif / xdif));
        }
        return {xsp:xsp,ysp:ysp};
    },
    //子弹结束
    over : function(enemy){
        //判断子弹是因为击中敌人而结束
        if(enemy){
            var effer = {},bType = BulletType[this.type]["level_"+this.level];
            //设置子弹的效果
            if(bType.forzen){
                effer = {effer:"frozen",num:bType.forzen};
            }
            else if(bType.steal){
                effer = {effer:"steal",num:bType.steal};
            }
            else if(bType.kill){
                effer = {effer:"kill",num:bType.kill};
            }
            else effer = {effer:"nomal"};
            //较少鸡蛋的生命
            enemy.reduceLife(bType.hurt,effer);
        }
        //移除子弹
        Game.bulletList.remove(this);
        //设置穿刺子弹的击中敌人列表为空
        this.lineEnemy = null;
        return false;
    },
    //穿刺子弹的伤害敌人
    lineHurt : function(enemy){
        enemy.reduceLife(BulletType[this.type]["level_"+this.level].hurt,{effer:"normal"});
    }
}
//子弹类型
var BulletType = [
    {
        level_1:{
            hurt:10,steal:0
        },
        level_2:{
            hurt:12,steal:0
        },
        level_3:{
            hurt:12,steal:1
        }
    },
    {
        level_1:{
            hurt:5,forzen:3000
        },
        level_2:{
            hurt:8,forzen:4000
        },
        level_3:{
            hurt:10,forzen:4000
        }
    },
    {
        level_1:{
            hurt:12
        },
        level_2:{
            hurt:15
        },
        level_3:{
            hurt:20
        }
    },
    {
        level_1:{
            hurt:100
        },
        level_2:{
            hurt:200
        },
        level_3:{
            hurt:300
        }
    },
    {
        level_1:{
            hurt:15,kill:5
        },
        level_2:{
            hurt:20,kill:8
        },
        level_3:{
            hurt:30,kill:10
        }
    }
]
//更新所有子弹信息
function updateBullet(){
    var bullet;
    for(var i=0,l=Game.bulletList.length;i<l;i++){
        bullet = Game.bulletList[i];
        if(!bullet)continue;
        bullet.update(Game.enemyList);
    }
}
//画出所有子弹
function drawBullet(){
    var bullet;
    for(var i=0,l=Game.bulletList.length;i<l;i++){
        bullet = Game.bulletList[i];
        if(!bullet)continue;
        bullet.draw();
    }
}



image.png

image.png

image.png


Enemy.js


image.png


image.png

tower.js新增update函数

//更新塔的信息
    update : function(enemyList){
        //判断冷却时间
        if(this.cd > 0){
            this.cd -= 1;
            return false;
        }
        var towerInfo = TowerType[this.type]["level_"+this.level],
            canShot = towerInfo.bullet,
            enemy;
        this.cd = towerInfo.cd;
        //遍历敌人
        for(var i=0,l=enemyList.length;i<l;i++){
            enemy = enemyList[i];
            if(!enemy)continue;
            //判断敌人是否在塔的攻击范围内
            if(T.rectInCircle(enemy,{x:this.x+25,y:this.y+25,radius:towerInfo.scope})){
                //可发送的子弹数减少
                canShot -= 1;
                //新增一个子弹,加入到子弹列表中
                Game.bulletList.push(new Bullet(Game.canvasList.main,Game.imgList.bullet_img,this.type,enemy,this.level,this.x+20,this.y+20,5,5));
                //如果可用子弹没了,退出
                if(canShot <= 0)break;
            }
        }
    }


项目源码:


项目源码

目录
相关文章
|
4天前
|
JavaScript 前端开发 测试技术
探索现代JavaScript开发的最佳实践
本文探讨了现代JavaScript开发中的最佳实践,涵盖ES6+特性、现代框架使用、模块化与代码分割、测试驱动开发、代码质量与性能优化、异步编程、SPA与MPA架构选择、服务端渲染和静态站点生成等内容,旨在帮助开发者提升代码质量和开发效率。
|
8天前
|
Web App开发 JavaScript 前端开发
深入浅出Node.js后端开发
【10月更文挑战第36天】本文将引导您探索Node.js的世界,通过实际案例揭示其背后的原理和实践方法。从基础的安装到高级的异步处理,我们将一起构建一个简单的后端服务,并讨论如何优化性能。无论您是新手还是有经验的开发者,这篇文章都将为您提供新的视角和深入的理解。
|
13天前
|
Web App开发 存储 JavaScript
深入浅出Node.js后端开发
【10月更文挑战第31天】本文将引导你进入Node.js的奇妙世界,探索其如何革新后端开发。通过浅显易懂的语言和实际代码示例,我们将一起学习Node.js的核心概念、搭建开发环境,以及实现一个简单但完整的Web应用。无论你是编程新手还是希望拓展技术的开发者,这篇文章都将为你打开一扇通往高效后端开发的大门。
|
9天前
|
运维 监控 JavaScript
鸿蒙next版开发:分析JS Crash(进程崩溃)
在HarmonyOS 5.0中,JS Crash指未处理的JavaScript异常导致应用意外退出。本文详细介绍如何分析JS Crash,包括异常捕获、日志分析和典型案例,帮助开发者定位问题、修复错误,提升应用稳定性。通过DevEco Studio收集日志,结合HiChecker工具,有效解决JS Crash问题。
28 4
|
14天前
|
Web App开发 JavaScript 前端开发
深入浅出Node.js后端开发
【10月更文挑战第30天】本文将通过一个Node.js的简单示例,引导你进入Node.js的世界。我们将从基础概念讲起,然后一步步深入到代码实现,最后总结Node.js在后端开发中的优势和应用场景。无论你是前端开发者还是后端新手,这篇文章都将为你打开一扇了解Node.js的大门。
28 2
|
11天前
|
Web App开发 JavaScript 前端开发
探索后端开发:Node.js与Express的完美结合
【10月更文挑战第33天】本文将带领读者深入了解Node.js和Express的强强联手,通过实际案例揭示它们如何简化后端开发流程,提升应用性能。我们将一起探索这两个技术的核心概念、优势以及它们如何共同作用于现代Web开发中。准备好,让我们一起开启这场技术之旅!
26 0
|
18天前
|
JavaScript 前端开发 安全
深入浅出Node.js后端开发
【10月更文挑战第26天】在这篇文章中,我们将一起探索Node.js的奇妙世界。不同于传统的Java或Python,Node.js以其异步非阻塞I/O和事件驱动的特性,在后端开发领域独树一帜。无论你是初学者还是资深开发者,这篇文章都将为你提供新的视角和思考。从基础概念到实际应用,我们一步步深入Node.js的世界,让你了解其不仅仅是JavaScript运行环境那么简单。
|
4月前
|
JavaScript Java 测试技术
基于springboot+vue.js+uniapp的客户关系管理系统附带文章源码部署视频讲解等
基于springboot+vue.js+uniapp的客户关系管理系统附带文章源码部署视频讲解等
96 2