zoukankan      html  css  js  c++  java
  • JavaScript小游戏--2048(PC端)

    1.初始化棋局

    $(document).ready(function() {
        prepare_for_mobile();   //适配移动端
        new_game();
    });

    2.开始新游戏

    function new_game() {
        back = [];
        //初始化棋盘
        init();
        //在随机两个格子生成数字
        generate_one_number();
        generate_one_number();
    }

    3.初始化

    function init() {
        for (var i = 0; i < 4; i++) {
            for (var j = 0; j < 4; j++) {
                var grid_cell = $('#grid_cell_' + i + '_' + j);
                grid_cell.css('top', get_pos_top(i, j) + 'rem');//每个格子距离顶部和左边的距离;
                grid_cell.css('left', get_pos_left(i, j) + 'rem');
            }
        }
        for (var i = 0; i < 4; i++) {
            board[i] = new Array();
            has_conflicted[i] = new Array();
            for (var j =0; j < 4; j++) {
                board[i][j] = 0;//生成二维数组,每个格子的数字初始化为0;
                has_conflicted[i][j] = false;
            }
        }
    
    
        update_board_view();
        score = 0;
        update_score(score);
    }

    4.更新棋局(改数字大小、数字背景、设置数字宽度、高度、位置)

     数字为0、 数字大于100、数字大于1024获取不同的数字大小及背景;

    function update_board_view() {
        $('.number_cell').remove();//??number_cell
        for (var i = 0; i < 4; i++) {
            for (var j = 0; j < 4; j++) {
                $('#grid_container').append('<div class="number_cell" id="number_cell_' + i + '_' + j + '"></div>');
                var number_cell = $('#number_cell_' + i + '_' + j);
                if (board[i][j] == 0) {
                    number_cell.css({
                        'width': '0',//宽度、高度为0
                        'height': '0',
                        'top': get_pos_top(i, j) + 2.5 + 'rem',//这是每个数字的位置,其相对于每个格子的位置还要加2.5rem(每个格子高度、宽度为5rem)
                        'left': get_pos_left(i, j) + 2.5 +'rem',
                        'line-height': '5rem',
                        'font-size': '3rem'
                    });
    
                }else if(board[i][j] >= 1024){
                    console.log(board[i][j]);
                    number_cell.css({
                        'width': '5rem',
                        'height': '5rem',
                        'top': get_pos_top(i, j) + 'rem',
                        'left': get_pos_left(i, j) + 'rem',
                        'background-color': get_number_background_color(board[i][j]),//改变背景颜色
                        'color': get_number_color(board[i][j]),//改变数字颜色
                        'line-height': '5rem',
                        'font-size': '2rem'//字体变小
                    });
                    number_cell.text(board[i][j]);
                }else if(board[i][j] >= 100){
                    number_cell.css({
                        'width': '5rem',
                        'height': '5rem',
                        'top': get_pos_top(i, j) + 'rem',
                        'left': get_pos_left(i, j) + 'rem',
                        'background-color': get_number_background_color(board[i][j]),
                        'color': get_number_color(board[i][j]),
                        'line-height': '5rem',
                        'font-size': '2.5rem'
                    });
                    number_cell.text(board[i][j]);
                }else {//小于100
                    number_cell.css({
                        'width': '5rem',
                        'height': '5rem',
                        'top': get_pos_top(i, j) + 'rem',
                        'left': get_pos_left(i, j) + 'rem',
                        'background-color': get_number_background_color(board[i][j]),
                        'color': get_number_color(board[i][j]),
                        'line-height': '5rem',
                        'font-size': '3rem'
                    });
                    number_cell.text(board[i][j]);//.text()方法设置或返回被选元素的文本内容
                }
                has_conflicted[i][j] = false;
            }
        }
    
    
    }

    5.随机在一个格子生成数字

    function generate_one_number() {
        if (nospace(board)) {//若nospace(board)返回true,说明没有空格子;
            return false;
        }
        //随机一个位置
        var randx = parseInt(Math.floor(Math.random() * 4));
        var randy = parseInt(Math.floor(Math.random() * 4));//在124行调用
        var time = 0;
        while (time < 50) {//time??50??
            if (board[randx][randy] == 0) {//看看随机选的位置上数字是否为0,为0则可用,跳出循环,不为0,则重新找位置。
                break;
            }
            randx = parseInt(Math.floor(Math.random() * 4));//当board[randx][randy]不为0,则重新选位置。
            randy = parseInt(Math.floor(Math.random() * 4));
            time++;
        }
        if (time == 50) {//如果连续找了50次都没找到数字为0的位置,则按二维数组挨个去找,直到找到第一个数字为0的格子;
            for (var i = 0; i < 4; i++) {
                for (var j = 0; j < 4; j++) {
                    if (board[i][j] == 0) {
                        randx = i;
                        randy = j;
                    }
                }
            }
        }
        //随机一个数字
        var rand_number = Math.random() < 0.5 ? 2 : 4;
        //在随机位置显示随机数字
        board[randx][randy] = rand_number;
        show_number_with_animation(randx, randy, rand_number);//动画显示数字格子
        save_status(board); //每生产一个数字,保存状态(以便撤回),模拟快照,每一步生成一个对象,调用support.js中函数
        return true;
    }

    6.监听键盘的上下左右移动

    keyCode(键码),指定与引发事件的键关联的Unicode的键码。此属性旨在与所述的onkeydown,的onkeyup和onkeypress事件的事件中使用。

    键盘上下左右 方向键的键码(keyCode)是38、40、37和39

    $(document).keydown(function(event) {
        if ($('#score').text() == success_string) {//一按下键盘首先判断是否成功。324行判断数字是否到达2048,若等于2048则调用showanimation.js中的更新分数函数;
            new_game();
            return;
        }
        switch (event.keyCode) {//键盘上下左右 方向键的键码(keyCode)是38、40、37和39
            case 37: //left
                event.preventDefault();
                if (move_left()) {
                    setTimeout('generate_one_number()', 200);//在指定的毫秒数后调用函数或计算表达式,setTimeout() 只执行 code 一次;
                    setTimeout('is_gameover()', 300);
                }
                break;
            case 38: //up
                event.preventDefault();
                if(move_up()){
                    setTimeout('generate_one_number()',200);
                    setTimeout('is_gameover()', 300);
                }
                break;
            case 39: //right
                event.preventDefault();
                if(move_right()){
                    setTimeout('generate_one_number()',200);
                    setTimeout('is_gameover()', 300);
                }
                break;
            case 40: //down
                event.preventDefault();
                if (move_down()){
                    setTimeout('generate_one_number()',200);
                    setTimeout('is_gameover()', 300);
                }
                break;
            default:
                break;
        }
    });

    7.向左移动 (向右移动、向上移动、向下移动类似)

    function move_left() {//判断能否左移,调用support.js
        if (!can_move_left(board)) {
            return false;
        }
        //move left
        for (var i = 0; i < 4; i++) {
            for (var j = 1; j < 4; j++) {  //j=1
                if (board[i][j] != 0) {
                    for (var k = 0; k < j; k++) {
                        if (board[i][k] == 0 && no_block_horizontal(i, k, j, board)) {//(i,k)为0且(i,k)到(i,j)之间都是0
                            show_move_animation(i, j, i, k);//格子移动时有动画效果,从(i,j)到(i,k)
                            board[i][k] = board[i][j];
                            board[i][j] = 0;
                            break;//跳出的只是199行的for循环,外面还有两层循环
                        } else if (board[i][k] == board[i][j] && no_block_horizontal(i, k, j, board) && !has_conflicted[i][k]) {//可合并的情况,(i,k)和(i,j)数字相等且(i,k)到(i,j)之间都是0
                            show_move_animation(i, j, i, k);
                            board[i][k] += board[i][j];
                            board[i][j] = 0;
                            //add score
                            score += board[i][k];
                            update_score(score);
                            has_conflicted[i][k] = true;//??
                            break;
                        }
                    }
                }
            }
        }
        setTimeout('update_board_view()', 200);
        return true;
    }

    8.判断游戏是否成功

    function is_gameover() {
        for(var i=0; i<4; i++){
            for(var j=0; j<4; j++){
                if (board[i][j] == 2048){
                    update_score(success_string);
                    return;
                }
            }
        }
        if (nospace(board) && nomove(board)) {//没有空格子,且各个方向不能移动合并;
            gameover();
        }
    }

    function gameover() {
       alert("走投无路啦");
    }

    9.保存状态

    function save_status(board) {
        var o = new Object();
        var n = 1;
        for(var i=0; i<4; i++){
            for(var j=0; j<4; j++){
                o[n] = score+ ',' + board[i][j];//每个o长度为16
                n ++;
            }
        }
        back.push(o);   //调用一次save_status(board),back长度加1.
    }

    10.撤销状态(回退)

    function come_back() {
        if (back.length<=2){//back数组长度小于2
            return;
        }
        var o = back[back.length-2];//结合support.js中保存状态的函数理解,back数组长度减2,back[]取减2后的位置上的值,(每生成一个数字会保存一次状态,back数组长度加1)
        var n = 1;
        //o[n]遍历对象
        for (var i = 0; i < 4; i++) {
            for (var j = 0; j < 4; j++) {//把存的score和board[i][j]取出
                board[i][j]  = parseInt(o[n].substr(o[n].lastIndexOf(',')+1));//lastIndexOf从右向左出现某个字符或字符串的首个字符索引值,该索引值加1,即可去到,的后半部分
                score = parseInt(o[n]);//逗号前面的
                n++;
            }
        }
        update_score(score);
        back.pop();
        setTimeout('update_board_view()',200);
    }
  • 相关阅读:
    JS保留小数点(四舍五入、四舍六入)实例
    HTML5 本地存储 localStorage、sessionStorage 的遍历、存储大小限制处理
    TCP/IP 网络编程(五)
    【IOS】mac终端运行.sh文件总是提示permission denied
    一扫天下——ZXing使用全解析
    《学习bash》笔记--进程处理
    VB断点调试
    poj 2506 Tiling(java解法)
    策略模式实战之优惠方式
    HTML 5 音频Audio
  • 原文地址:https://www.cnblogs.com/yujihang/p/6691004.html
Copyright © 2011-2022 走看看