zoukankan      html  css  js  c++  java
  • d3.js 制作简单的俄罗斯方块

    d3.js是一个不错的可视化框架,同时对于操作dom也是十分方便的。今天我们使用d3.js配合es6的类来制作一个童年小游戏--俄罗斯方块。话不多说先上图片。

    1. js tetris类

    由于方法拆分的比较细所以加上了一些备注(这不是我的风格!)

    const graphMap = [
        {
            name: '倒梯形',
            position: [[0,4],[1,3],[1,4],[1,5]],
            rotate: [[[0,0],[-2,0],[-1,-1],[0,-2]],[[1,0],[1,2],[0,1],[-1,0]],[[-1,0],[1,0],[0,1],[-1,2]],[[0,0],[0,-2],[1,-1],[2,0]]],
            color: '#D7DF01'
        },
        {
            name: '一字型',
            position: [[0,3],[0,4],[0,5],[0,6]],
            rotate: [[[-1,1],[0,0],[1,-1],[2,-2]],[[1,2],[0,1],[-1,0],[-2,-1]],[[2,-2],[1,-1],[0,0],[-1,1]],[[-2,-1],[-1,0],[0,1],[1,2]]],
            color: '#0000FF'
        },
        {
            name: '正方形',
            position: [[0,4],[0,5],[1,4],[1,5]],
            rotate: [[[0,0],[0,0],[0,0],[0,0]],[[0,0],[0,0],[0,0],[0,0]],[[0,0],[0,0],[0,0],[0,0]],[[0,0],[0,0],[0,0],[0,0]]],
            color: '#FF0000'
        },
        {
            name: 'Z字型',
            position: [[0,3],[0,4],[1,4],[1,5]],
            rotate: [[[0,1],[1,0],[0,-1],[1,-2]],[[1,1],[0,0],[-1,1],[-2,0]],[[1,-2],[0,-1],[1,0],[0,1]],[[-2,0],[-1,1],[0,0],[1,1]]],
            color: '#800080'
        },
        {
            name: '反Z字型',
            position: [[1,3],[1,4],[0,4],[0,5]],
            rotate: [[[-1,1],[0,0],[1,1],[2,0]],[[0,1],[-1,0],[0,-1],[-1,-2]],[[2,0],[1,1],[0,0],[-1,1]],[[-1,-2],[0,-1],[-1,0],[0,1]]],
            color: '#FFA500'
        },
        {
            name: 'L字型',
            position: [[1,3],[0,3],[0,4],[0,5]],
            rotate: [[[-1,1],[0,2],[1,1],[2,0]],[[0,1],[1,0],[0,-1],[-1,-2]],[[1,-1],[0,-2],[-1,-1],[-2,0]],[[0,-1],[-1,0],[0,1],[1,2]]],
            color: '#90EE90'
        },
        {
            name: '反L字型',
            position: [[0,3],[0,4],[0,5],[1,5]],
            rotate: [[[-1,2],[0,1],[1,0],[0,-1]],[[2,0],[1,-1],[0,-2],[-1,-1]],[[1,-2],[0,-1],[-1,0],[0,1]],[[-2,0],[-1,1],[0,2],[1,1]]],
            color: '#AEEBFF'
        }
    ]
    class Tetris {
        constructor() {
            this._grid = [];
            this._rows = 18;
            this._cols = 10;
            this._div = 33;
            this._nextDiv = 15;
            this._duration = 1000;
            this._width = this._div * this._cols;
            this._height = this._div * this._rows;
            this._svg = null;
            this._nextSvg = null;
            this._timeout = null;
            this._time = null;
            this._showGrid = false;
            this._haveArray = [];
            this._curtArray = [];
            this._colors = '';
            this._rotateIndex = 0;
            this._rotateArray = [];
            this._fixedColor = '#666';
            this._nextNumber = 0;
            this._graphMap = graphMap;
            this._level = 1;
            this._levelLimit = [0,20,50,90,140,200,270,350,440,540,650,770,900,1040,1190,1350,1520];
            this._score = 0;
            this._timeNumber = 0;
            this.initSvg();
            this.initNextSvg();
            this.addKeyListener();
        }
        initSvg() {
            this._svg = d3.select('.svg-container')
                .append('svg')
                .attr('width', this._width)
                .attr('height', this._height)
                .attr('transform', 'translate(0, 4)')
        }
        initNextSvg() {
            this._nextSvg = d3.select('.next')
            .append('svg')
            .attr('width', 100)
            .attr('height', 60)
        }
        toggleGrid() {
            if(this._showGrid) {
                this._showGrid = false;
                d3.select('g.grid').remove();
            } else {
                this._showGrid = true;
                this._grid = this._svg.append('g')
                .attr('class', 'grid')
                this._grid.selectAll('line.row')
                .data(d3.range(this._rows))
                .enter()
                .append('line')
                .attr('class', 'row')
                .attr('x1', 0)
                .attr('y1', d => d * this._div)
                .attr('x2', this._width)
                .attr('y2', d => d * this._div)
                this._grid.selectAll('line.col')
                .data(d3.range(this._cols))
                .enter()
                .append('line')
                .attr('class', 'col')
                .attr('x1', d => d * this._div)
                .attr('y1', 0)
                .attr('x2', d => d * this._div)
                .attr('y2', this._height)
            }
        }
        addKeyListener() {
            d3.select('body').on('keydown', () => {
                switch (d3.event.keyCode) {
                    case 37:
                        this.goLeft();
                    break;
                    case 38:
                        this.rotate();
                        break;
                    case 39:
                        this.goRight();
                        break;
                    case 40:
                        this.goDown();
                        break;
                    case 32:
                        console.log('空格');
                        break;
                    case 80:
                        console.log('暂停');
                        break;
                    default:
                        break;
                }
            })
        }
        //设置运动图形 如果仍有掉落空间则继续掉落 反之调用setHaveArray
        initGraph() {
            this.renderGraph();
            this._timeout = setTimeout(() => {
                if(this.canDown()) {
                    this.downArray();
                    this.initGraph();
                } else {
                    this.setHaveArray();
                    if(!this.gameOver()) {
                        this.randomData();
                        this.nextGraphNumber();
                        this.initGraph();
                    } else {
                        clearTimeout(this._time);
                        d3.select('#modal').style('top', '0px')
                    }
                }
            }, this._duration * (1 - ((this._level - 1) / this._levelLimit.length) / 2))
        }
        //渲染图形
        renderGraph() {
            this._svg.selectAll('rect.active').remove();
            this._svg.selectAll('rect.active')
            .data(this._curtArray)
            .enter()
            .append('rect')
            .attr('class', 'active')
            .attr('x', d => this._div * d[1] + 1)
            .attr('y', d => this._div * d[0] + 1)
            .attr('width', this._div - 3)
            .attr('height', this._div - 3)
            .attr('stroke', this._color)
            .attr('stroke-width', 2)
            .attr('fill', this._color)
            .attr('fill-opacity', 0.5)
        }
        //设置掉落后的数组,并清除运动的图形 重置状态
        setHaveArray() {
            this._curtArray.forEach(d => this._haveArray.push(d));
            this._svg.selectAll('rect.active').attr('class', 'fixed').attr('fill', this._fixedColor).attr('fill-opacity', 0.5).attr('stroke', this._fixedColor);
            this._rotateIndex = 0;
            this.clearLines();
        }
        //检测有满列 然后消除
        clearLines() {
            let clearLinesArr = [];
            let allRowsObj = {};
            let temp = [];
            let arr = this._haveArray.map(d => d[0]);
            arr.forEach(d => {
                if(allRowsObj.hasOwnProperty(d)) {
                    allRowsObj[d] ++
                } else {
                    allRowsObj[d] = 1;
                }
            })
            for(var i in allRowsObj) {
                if(allRowsObj[i] == this._cols) {
                    clearLinesArr.push(i)
                }
            }
            if(clearLinesArr.length != 0) {
                this.setScoreAndLevel(clearLinesArr.length);
                this._haveArray = this._haveArray.filter(a => !clearLinesArr.some(b => b == a[0]));
                this._haveArray = this._haveArray.map(d => [this.downSome(d[0],clearLinesArr), d[1]])
                this._svg.selectAll('rect.fixed').remove();
                this._svg.selectAll('rect.fixed').data(this._haveArray)
                .enter()
                .append('rect')
                .attr('class', 'fixed')
                .attr('x', d => this._div * d[1] + 1)
                .attr('y', d => this._div * d[0] + 1)
                .attr('width', this._div - 3)
                .attr('height', this._div - 3)
                .attr('stroke', this._fixedColor)
                .attr('stroke-width', 2)
                .attr('fill', this._fixedColor)
                .attr('fill-opacity', 0.5)
            }
        }
        //消除时 判断下落层数
        downSome(c, arr) {
            let num = 0;
            arr.forEach(d => {
                if(c < d) {
                    num ++;
                }
            })
            return num + c;
        }
        //设置等级和分数
        setScoreAndLevel(num) {
            switch(num) {
                case 1:
                    this._score = this._score + 1;
                    break;
                case 2:
                    this._score = this._score + 3;
                    break;
                case 3:
                    this._score = this._score + 6;
                    break;
                case 4:
                    this._score = this._score + 10;
                default:
                    break;
            }
            for(var i=0; i<this._levelLimit.length; i++) {
                if(this._score <= this._levelLimit[i]) {
                    this._level = i + 1;
                    break;
                }
            }
            d3.select('#score').html(this._score);
            d3.select('#level').html(this._level);
        }
        //左移动
        goLeft() {
            if(this.canLeft()) {
                this.leftArray();
                this.renderGraph();
            }
        }
        //右移动
        goRight() {
            if(this.canRight()) {
                this.rightArray();
                this.renderGraph();
            }
        }
        //旋转
        rotate() {
            if(this.canRotate()) {
                this.rotateArray();
                this.renderGraph();
            }
        }
        //下移动
        goDown() {
            if(this.canDown()) {
                this.downArray();
                this.renderGraph();
            }
        }
        //下落更新数组
        downArray() {
            this._curtArray = this._curtArray.map(d => {
                return [d[0] + 1, d[1]]
            })
        }
        //左移更新数组
        leftArray() {
            this._curtArray = this._curtArray.map(d => {
                return [d[0], d[1] - 1]
            })
        }
        //右移更新数组
        rightArray() {
            this._curtArray = this._curtArray.map(d => {
                return [d[0], d[1] + 1]
            })
        }
        //旋转更新数组
        rotateArray() {
            let arr = this._rotateArray[this._rotateIndex];
            this._curtArray = this._curtArray.map((d,i) => {
                return [d[0] + arr[i][0], d[1] + arr[i][1]]
            })
            this._rotateIndex = (this._rotateIndex + 1) % 4;
        }
        //判断是否可以下落
        canDown() {
            let max = 0;
            let status = true;
            let nextArr = this._curtArray.map(d => {
                if(d[0] + 1 > max) {
                    max = d[0];
                }
                return [d[0] + 1, d[1]]
            });
            nextArr.forEach(d => {
                this._haveArray.forEach(item => {
                    if(item[0] == d[0] && item[1] == d[1]) {
                        status = false;
                    }
                })
            })
            if(!status || max > 16) {
                return false;
            } else {
                return true;
            }
        }
        //判断是否可以左移
        canLeft() {
            let min = this._cols;
            let status = true;
            let nextArr = this._curtArray.map(d => {
                if(d[1] - 1 < min) {
                    min = d[1];
                }
                return [d[0], d[1] - 1]
            })
            nextArr.forEach(d => {
                this._haveArray.forEach(item => {
                    if(item[0] == d[0] && item[1] == d[1]) {
                        status = false;
                    }
                })
            })
            if(!status || min <= 0) {
                return false;
            } else {
                return true;
            }
        }
        //判断是否可以右移
        canRight() {
            let max = 0;
            let status = true;
            let nextArr = this._curtArray.map(d => {
                if(d[1] + 1 > max) {
                    max = d[1];
                }
                return [d[0], d[1] + 1]
            })
            nextArr.forEach(d => {
                this._haveArray.forEach(item => {
                    if(item[0] == d[0] && item[1] == d[1]) {
                        status = false;
                    }
                })
            })
            if(!status || max > this._cols - 2) {
                return false;
            } else {
                return true;
            }
        }
        //判断可以变形
        canRotate() {
            let max = 0;
            let min = this._cols;
            let status = true;
            let arr = this._rotateArray[this._rotateIndex];
            let nextArr = this._curtArray.map((d,i) => {
                if(d[1] + 1 > max) {
                    max = d[1];
                }
                if(d[1] - 1 < min) {
                    min = d[1];
                }
                return [d[0] + arr[i][0], d[1] + arr[i][1]]
            })
            if(!status || max > this._cols - 1 || min < 0) {
                return false;
            } else {
                return true;
            }
        }
        //判断游戏结束
        gameOver() {
            let status = false;
            this._haveArray.forEach(d => {
                if((d[0] == 0 && d[1] == 3) || (d[0] == 0 && d[1] == 4) || (d[0] == 0 && d[1] == 5) || (d[0] == 0 && d[1] == 6)) {
                    status = true;
                }
            })
            return status;
        }
        //随机生成图形块
        randomData() {
            this._curtArray = this._graphMap[this._nextNumber].position;
            this._color = this._graphMap[this._nextNumber].color;
            this._rotateArray = this._graphMap[this._nextNumber].rotate;
        }
        //预设下一个图形展示
        nextGraphNumber() {
            let rand = [0,0,1,1,2,2,3,4,5,6];
            this._nextNumber = rand[Math.floor(Math.random() * 10000) % 10];
            this._nextSvg.selectAll('rect.ne').remove();
            this._nextSvg.selectAll('rect.ne')
                .data(this._graphMap[this._nextNumber].position)
                .enter()
                .append('rect')
                .attr('class', 'ne')
                .attr('x', d => this._nextDiv * (d[1] - 1) + 1)
                .attr('y', d => this._nextDiv * (d[0] + 1) + 1)
                .attr('width', this._nextDiv - 3)
                .attr('height', this._nextDiv - 3)
                .attr('stroke', this._graphMap[this._nextNumber].color)
                .attr('stroke-width', 2)
                .attr('fill', this._graphMap[this._nextNumber].color)
                .attr('fill-opacity', 0.5)
        }
        //初始化数据
        initData() {
            this._haveArray = [];
            this._level = 1;
            this._score = 0;
            this._timeNumber = 0;
            this._svg.selectAll('rect').remove();
            d3.select('#score').html(0);
            d3.select('#level').html(1);
            d3.select('#time').html(0);
        }
        //开始时间
        initTime() {
            this._time = setInterval(() => {
                this._timeNumber ++;
                d3.select('#time').html(this._timeNumber);
            },1000)
        }
        //开始游戏
        startGame() {
            this.initData();
            this.randomData();
            this.nextGraphNumber();
            this.initGraph();
            this.initTime();
        }
    }

    2. css 代码

    * {
      padding: 0;
      margin: 0;
    }
    body {
      width: 480px;
      margin: 30px auto;
    }
    .svg-container {
      overflow: hidden;
      border: 5px solid rgba(0,0,0,0.2);
      width: 330px;
      position: relative;
      float: left;
    }
    #modal {
      position: absolute;
      top: 0px;
      background-color: white;
      border-bottom: 5px solid rgb(202,202,202);
      padding: 20px;
      width: 310px;
      text-align: center;
      z-index: 100;
      transition: 200ms linear;
    }
    #newGame {
      text-decoration: none;
      color: gray;
      font-size: 25px;
      cursor: pointer;
    }
    aside {
      position: relative;
      float: right;
    }
    aside .next {
      width: 100px;
      height: 60px;
      padding: 10px;
      border: 5px solid rgba(0,0,0,0.2);
      border-radius: 2px;
      margin-bottom: 10px;
    }
    aside .score {
      width: 100px;
      padding: 10px;
      border: 5px solid rgba(0,0,0,0.2);
      border-radius: 2px;
      color: gray;
    }
    aside .pause {
      color: gray;
      font-size: 12px;
      font-style: italic;
      padding-left: 3px;
      margin-top: 15px;
    }
    .row {
      stroke: lightgray;
    }
    .col {
      stroke: lightgray;
    }

    3. html代码

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>$Title$</title>
        <link rel="stylesheet" type="text/css" href="css/base.css"/>
        <script type="text/javascript" src="js/d3.v4.js"></script>
        <script type="text/javascript" src="js/base.js"></script>
    </head>
    <body>
        <div class="container">
            <div class="svg-container">
                <div id="modal" class="active">
                        <span id="newGame" onclick="newGame()">New Game</span>
                </div>
            </div>
            <aside>
                <div class="next"></div>
                <div class="score">
                    <table>
                        <tr>
                            <td>Level:</td>
                            <td id="level"></td>
                        </tr>
                        <tr>
                            <td>Score:</td>
                            <td id="score"></td>
                        </tr>
                        <tr>
                            <td>Time:</td>
                            <td id="time"></td>
                        </tr>
                    </table>
                </div>
                <div class="pause">
                    <input type="checkbox" onclick="toggleGrid()"/> 网格
                </div>
            </aside>
        </div>
    <script>
    var tetris = new Tetris();
    
    function toggleGrid() {
        tetris.toggleGrid()
    }
    function newGame() {
        document.getElementById('modal').style.top = '-100px';
        tetris.startGame()
    }
    </script>
    </body>
    </html>

    想预览或者下载demo的人请移步至原文

    原文地址 http://www.bettersmile.cn

  • 相关阅读:
    八款前端开发人员更轻松的实用在线工具
    HTML5中的Web Notification桌面通知(右下角提示)
    老司机程序员用到的各种网站整理
    JAVA变量存储
    关于JAVA中的前期绑定 后期绑定(动态绑定)
    i MySQL 查看约束,添加约束,删除约束
    final static
    MySQL alter语句
    MySQL 权限生效
    MySQL 用户权限管理
  • 原文地址:https://www.cnblogs.com/vadim-web/p/11492095.html
Copyright © 2011-2022 走看看