最新公告
  • 欢迎您光临起源地模板网,本站秉承服务宗旨 履行“站长”责任,销售只是起点 服务永无止境!立即加入钻石VIP
  • 使用vue3重构拼图游戏,真香!

    正文概述 掘金(刘小灰)   2021-01-22   475

    前言

    花了两天时间,重构了项目中的一个拼图小游戏(又名数字华容道),为了方便使用抽离成了独立组件,效果如下:

    使用vue3重构拼图游戏,真香!使用vue3重构拼图游戏,真香!

    线上体验

    源码地址在文章最后哦!

    主要重构点

    原有拼图游戏是通过开源代码加以改造,使用的是 vue2 。在实际项目使用一切正常,但还是存在以下痛点

    • 源代码臃肿,暴露的配置项不足,特备是和项目现有逻辑结合时体现的更加明显
    • 生成的游戏可能出现无解情况,为了避免无解,只好写死几种情况然后随机生成
    • 源代码是vue2版本,不支持vue3

    最后决定使用 vue3 重新实现拼图游戏,着重注意以下细节

    1. 组件使用起来足够简单
    2. 可以自定义游戏难度
    3. 支持图片和数组两种模式

    实现思路

    无论是拼图片还是拼数字,其原理都是要把原本打乱的数组移动成有序状态。网上也有很多实现数字华容的的算法,算法主要需要解决的就是如何生成一组 随机且有解 的数组,有的人可能有疑问,数组华容道还有可能无解吗?

    使用vue3重构拼图游戏,真香!

    如果生成的游戏像上面这样,那就是无解了。原理就像我们玩魔方一样,正常情况下不管我们打乱的多乱都可以还原,但是如果我们把 某几个块抠出来换换位置 ,那就可能还原不了了

    网上也有很多算法可以避免生成无解的情况,但是写的都比较高深,加上自己阅读算法的能力有限,最后决定按照自己的思路实现。

    我的思路其实比较简单,一句话总结就是逆向推理法,我们可以先从排列好的数组开始,然后随机的通过 移动 打乱其顺序,这样肯定可以保证生成的题目有解,同时可以根据打乱的步数来控制游戏的难度,真是个一举两得的idear。

    源码实现

    数据存放

    首先我考虑的是使用一个一维数组来存放数据

    let arr = [1,2,3,4,5,6,7,8,0]  // 0 代表空白
    

    但是这样会出现一个问题,就是移动的时候逻辑变的相当麻烦,因为一维数组只能记录数据并不能记录竖直方向的位置,这个时候我们自然就想到使用二维数组了,比如,当我们需要生成一个3*3的游戏时数据是这样的:

    let arr [
        [1,2,3],
        [4,5,6],
        [7,8,0]
    ]
    

    这样我们就可以通过下面来模拟x,y轴来表示每个数字的位置。比如0的位置是(2,2)6的位置是(1,2),如果我想移动6和0的位置,只需要把他们的坐标做调换即可。

    移动函数

    数字华容道最关键的交互就是用户点击那个块就移动哪个块,但是需要注意的是只有0附近的数组可以移动。下面我们先完成移动函数

      function move(x, y, moveX, moveY) {
        const num = state.arr[x][y];
        state.arr[x][y] = state.arr[moveX][moveY];
        state.arr[moveX][moveY] = num;
      }
    

    是不是很简单,其实就是把要移动的两个数的下标给交换下位置

    有了移动函数,我们就可以实现上,下,左,右的移动了

    // 上移动
      function moveTop(x, y) {
        if (x <= 0) return -1;
        //   开始交换位置
        const okx = x - 1;
        move(x, y, okx, y);
        return {
          x: okx,
          y,
        };
      }
      //下移动
      function moveDown(x, y) {
        if (x >= level - 1) return -1;
        const okx = x + 1;
        move(x, y, okx, y);
        return {
          x: okx,
          y,
        };
      }
      // 左移动
    
      function moveLeft(x, y) {
        if (y <= 0) return -1;
        const oky = y - 1;
        move(x, y, x, oky);
        return {
          x,
          y: oky,
        };
      }
    
      // 右移动
      function moveRight(x, y) {
        if (y >= level - 1) return -1;
        const oky = y + 1;
        move(x, y, x, oky);
        return {
          x,
          y: oky,
        };
      }
    
    

    现在我们再实现一个判断移动方向的方法,如下所示:

      function shouldMove(x, y) {
        //   判断向哪移动
        const { emptyX, emptyY } = seekEmpty();
        if (x === emptyX && y !== emptyY && Math.abs(y - emptyY) === 1) {
          // 说明在一个水平线上 可能是左右移动
          if (y > emptyY) {
            moveLeft(x, y);
          } else {
            moveRight(x, y);
          }
        }
        if (y === emptyY && x !== emptyX && Math.abs(x - emptyX) === 1) {
          // 说明需要上下移动
          if (x > emptyX) {
            moveTop(x, y);
          } else {
            moveDown(x, y);
          }
        }
      }
    

    if里面判断的意思是如果我们点击的块是空白快或者不是和空白快挨着的那个,那我们就不做任何处理

    生成游戏面板

    其实就是随机调用上移,下移,左移,右移函数,把数组打乱

      // 随机打乱
      function moveInit(diffic) {
        state.arr = creatArr(level);
        const num = diffic ? diffic : state.diffec;
        const fns = [moveTop, moveDown, moveLeft, moveRight];
        let Index = null;
        let fn;
        for (let i = 0; i < num; i++) {
          Index = Math.floor(Math.random() * fns.length);
          //   moveConsole(Index);
          fn = fns[Index](startX, startY);
          if (fn != -1) {
            const { x, y } = fn;
            startX = x;
            startY = y;
          }
        }
      }
    
    

    短短几个函数,就完成了核心逻辑,还有几个函数没有介绍到比如判断游戏完成,寻找空白块的位置,创建二维数组大家可自行阅读源码

    使用vue3重构

    以上逻辑貌似和vue3没什么关系,但是我们忽略了最重要的一点,就是 改变数组后,视图也就改变了 这不就是响应式吗,使用vue3后我们可以把关于游戏的所有逻辑放到一个js里面,大大减少了代码的耦合度

    const { arr, shouldMove, moveInit } = useCreateGame(
      gamedata.level,
      gamedata.difficulty,
      gameEndCallback
    );
    

    可能有的人会有疑问?把所有逻辑抽离出来这不是很正常的操作吗?难道用vue2的时候都不能抽离了?

    但是大家不要忘记了,我们的这个数组需要是响应式的,如果我们单独把逻辑抽离出来那我们在js文件里面改变数组还是响应式的吗

    但当我们使用vue3的composition-api时就可以再js文件中声明一个响应式变量,且在组件中使用时它还是响应式的

    
    export default function useCreateGame() {
    //声明一个响应式变量
    ...
      const state = reactive({
        arr: [],
      });
    ...
      return {
      ...toRefs(state)
      ...
      }
     }
    
    const { arr, shouldMove, moveInit } = useCreateGame(
      gamedata.level,
      gamedata.difficulty,
      gameEndCallback
    );
    // 这个时候 arr 还是响应式的
    

    这正是composition-api强大所在,有了composition-api我们可以任意组装我们的逻辑代码了

    关于vite2

    现在vite已经到了vite2版本,并且官方还在飞快迭代中,使用vite2创建的项目默认是可以使用setup新特性的,比如我们可以这样写:

    <template>
      <div>
        {{ name }}
      </div>
    </template>
    
    <script setup>
    import { ref } from "vue";
    const name = ref('"公众号码不停息"');
    </script>
    

    等价于这样写

    <template>
      <div>
        {{ name }}
      </div>
    </template>
    <script>
    import { ref } from "vue";
    export default {
      setup() {
        const name = ref("公众号码不停息");
        return {
          name,
        };
      },
    };
    </script>
    

    看着就简单了很多,并且在setup版本中vue又出了几个api,感兴趣的可以去官网看下,个人感觉还是挺香的。因为新语法还是实验性质的本次代码重构并未使用。

    源码地址

    源码地址: 数字华容道拼图游戏 欢迎start?

    最后

    你可能感兴趣:

    • 基于vue-router思考?实现一个简易版vue-router
    • 基于webpack打包多页应用,对前端工程化的思考

    如果帮助,欢迎点赞哦?


    起源地下载网 » 使用vue3重构拼图游戏,真香!

    常见问题FAQ

    免费下载或者VIP会员专享资源能否直接商用?
    本站所有资源版权均属于原作者所有,这里所提供资源均只能用于参考学习用,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担。更多说明请参考 VIP介绍。
    提示下载完但解压或打开不了?
    最常见的情况是下载不完整: 可对比下载完压缩包的与网盘上的容量,若小于网盘提示的容量则是这个原因。这是浏览器下载的bug,建议用百度网盘软件或迅雷下载。若排除这种情况,可在对应资源底部留言,或 联络我们.。
    找不到素材资源介绍文章里的示例图片?
    对于PPT,KEY,Mockups,APP,网页模版等类型的素材,文章内用于介绍的图片通常并不包含在对应可供下载素材包内。这些相关商业图片需另外购买,且本站不负责(也没有办法)找到出处。 同样地一些字体文件也是这种情况,但部分素材会在素材包内有一份字体下载链接清单。
    模板不会安装或需要功能定制以及二次开发?
    请QQ联系我们

    发表评论

    还没有评论,快来抢沙发吧!

    如需帝国cms功能定制以及二次开发请联系我们

    联系作者

    请选择支付方式

    ×
    迅虎支付宝
    迅虎微信
    支付宝当面付
    余额支付
    ×
    微信扫码支付 0 元