• 六边形拼图游戏之给提示


    此博客链接:https://www.cnblogs.com/ping2yingshi/p/14483090.html

    六边形拼图

    1.说明

    1.游戏玩法

     界面上显示一张被3*3六边形分割后打乱顺序的图片如图1.1打乱顺序图所示。(图片部分图片显示蓝色,称为蓝色图片),用户移动图片时,只能移动蓝色图片附近的图片,只需要点击图片,使图片和粉红色图片交换,最后把打乱顺序后的图片还原为原图片如图1.2原图所示。

                               1.1打乱顺序图

     

                                      1.2原图

    2.编程语言

    小程序

    2.技术原型

    (1)显示六边形

    (2)分割图片

    (3)移动图片

    (4)启发式搜索算法

    3.需求

    (1)在界面固定坐标处显示几张六边形图片。

    (2)打乱图片顺序。

    (3)交换图片。

    (4)当用户点击提示时,提示用户下一步移动哪一块儿图片。

    (5)判断用户移动图片是否和原图片一致。

    4.思路

    4.1说明

    使用代码分割图片,使每个图片显示六边形,把图片的背景位置存放到一个数组中,给每个图片绑定一个编号。使用启发式搜索算法,计算成功时需要的步骤,获取下一步的坐标,判断需要移动的图片,给出移动提示。

    4.2思路

    1.显示图片:把一张图片以六边形的形式展示。

    2.分割图片:把一张大的图片分割成多个小的六边形。

    3.获取用户点击事件:获取用户点击事件,判断图片是否和蓝色图片相邻,如果相邻则交换蓝色图片和选的图片的坐标。

    4.通关:通过判断图片编号和数组下标是否相等来判断图片是否还原了。

    5.提示:当用户点击提示时,给出移动提示。

    5.关键技术和核心代码

    5.1显示六边形

    5.1.1说明

    以六边形展示图片。

    5.1.2技术

    同一宽高的图片以不同固定角度叠加到一起,只取重合部分,视角效果会感觉像是一张六边形图片。

    同一张图片在同一位置以不同角度显示三次,只显示重合位置,不重合位置隐藏。

     

    5.1.3代码

    数据结构

     backup_liubianxing: [{
            className: 'pic1-1',
            flag: 0,
            fragment_no: 0,
          },
          {
            className: 'pic1-2',
            flag: 0,
            fragment_no: 1,
          },
          {
            className: 'pic1-3',
            flag: 0,
            fragment_no: 2,
          },
          {
            className: 'pic1-4',
            flag: 0,
            fragment_no: 3,
          },
          {
            className: 'pic2-1',
            flag: 0,
            fragment_no: 4,
          },
          {
            className: 'pic2-2',
            fragment_no: 5,
          },
          {
            className: 'pic2-3',
            fragment_no: 6,
    
          },
          {
            className: 'pic2-4',
            fragment_no: 7,
          },
          {
            className: 'pic4-4',
            fragment_no: 8,
            flag:1
          },
        ],

     wxml代码

    <view class="box">
        <block wx:for="{{liubianxing_pic}}" wx:for-index="row_index" wx:for-item="out_item">
            <view class="line{{row_index+1}}">
                <view class="pic-item" wx:for="{{out_item}}">
                    <view class="boxF">
                        <view class="boxS">
                            <view class="boxT {{item.className}}" data-row-index="{{row_index}}" data-col-index="{{index}}" bindtap="click_pic"></view>
                        </view>
                    </view>
                </view>
            </view>
        </block>
    </view>

    wxss代码

    .boxF,
    .boxS,
    .boxT {
         80px;
        height: 100px;
        overflow: hidden;
    }
    
    .boxF,
    .boxS {
        visibility: hidden;
    }
    
    .boxF {
        transform: rotate(120deg);
        float: left;
    }
    
    .boxS {
        transform: rotate(-60deg);
    }
    
    .boxT {
        transform: rotate(-60deg);
        /* background: no-repeat 50% center; */
        /* background-size: 125% auto; */
        visibility: visible;
        background-repeat: no-repeat;
        background-image: url();
    }
    
    .pic-item {
        float: left;
        margin-left: 4px;
        display: inline-block;
        overflow: hidden;
    }
    
    .pic-item:after {
        content: '';
        height: 0;
        line-height: 0;
        display: block;
        visibility: hidden;
        clear: both;
    }
    
    .pic1-1 {
        /* width = 80 */
        background-position: 0 0;
    }
    
    .pic1-2 {
        /* width = 80 */
        /* 1*width  */
        background-position: -80px 0;
    }
    
    .pic1-3 {
        /* width = 80 */
        /* 2*width  */
        background-position: -160px 0;
    }
    
    .pic1-4 {
        /* width = 80 */
        /* 3*width  */
        background-position: -240px 0;
    }
    
    .pic2-1 {
        /* width = 80 */
        /* height = 100 */
        /* width + width/2  */
        background-position: -40px -75px;
    }
    
    .pic2-2 {
        /* width = 80 */
        /* height = 100 */
        /* width + width/2  */
        background-position: -120px -75px;
    }
    
    .pic2-3 {
        /* width = 80 */
        /* height = 100 */
        /* width + width/2  */
        background-position: -200px -75px;
    }
    
    .pic2-4 {
        /* width = 80 */
        /* height = 100 */
        /* width + width/2  */
        background-position: -280px -75px;
    }
    
    .pic3-1 {
        /* width = 80 */
        background-position: 0px -150px;
    }
    
    .pic3-2 {
        /* width = 80 */
        /* 1*width  */
        background-position: -80px -150px;
    }
    
    .pic3-3 {
        /* width = 80 */
        /* 2*width  */
        background-position: -160px -150px;
    }
    
    .pic3-4 {
        /* width = 80 */
        /* 3*width  */
        background-position: -240px -150px;
    }
    
    .pic4-1 {
        /* width = 80 */
        /* height = 100 */
        /* width + width/2  */
        background-position: -40px -225px;
        background-color: rgb(191, 216, 223);
    }
    
    .pic4-2 {
        /* width = 80 */
        /* height = 100 */
        /* width + width/2  */
        background-position: -120px -225px;
        background-color: rgb(191, 216, 223);
    }
    
    .pic4-3 {
        /* width = 80 */
        /* height = 100 */
        /* width + width/2  */
        background-position: -200px -225px;
        background-color: rgb(191, 216, 223);
    }
    
    .pic4-4 {
        /* width = 80 */
        /* height = 100 */
        /* width + width/2  */
        background-position: -1000px -1000px;
        background-color: rgb(191, 216, 223);
    }
    View Code

    5.2找出下一步

    5.2.1说明

    用户点击提示时,给出下一步提示。需要先把从当前位置到完成拼图时,走的所有步骤记录下来。

    5.2.2技术

    启发式搜索算法

    5.2.2.1定义

    参见百度百科解释:https://baike.baidu.com/item/%E5%90%AF%E5%8F%91%E5%BC%8F%E6%90%9C%E7%B4%A2/8944170?fr=aladdin

    5.2.2.2 估价函数

    f(x)=g(x)+h(x)

    f(x)为估算函数

    g(x)为初始状态到当前状态的全部距离

    h(x)为当前状态到目标状态所走的步数

    5.2.2.3搜索过程

    (1)把初始节点S0放入到Open表中,f(S0)=g(S0)+h(S0);

    (2)如果Open表为空,则问题无解,失败退出;

    (3)把Open表中的第一个节点取出放入到Closed表,并记该节点为n;

    (4)考察节点是否为目标节点。如果是的话,则找到问题的解,成功退出;

    (5)如果节点n不可扩展,则转到第(2)步骤;

    (6)扩展节点n,生成子节点ni(i=1,2......)计算每一个子节点的估价值f(ni)(i=1,2......),并为每一个子节点设置指向父节点的指针,然后将这些子节点放入到Open表中;

    (7)根据各节点的估价函数值,对Open表中的全部节点按从小到大的顺序重新进行排序;

    (8)转第(2)步。 

    5.2.3代码

    坐标位置的数据结构

        this.place = [ //坐标位置
            [0, 0],
            [1, 0],
            [2, 0],
            [0, 1],
            [1, 1],
            [2, 1],
            [0, 2],
            [1, 2],
            [2, 2]
        ];

    两个坐标之间的曼哈顿距离

    distance: function (a, b) {
        let aa = this.getCoordinate(a);
        let bb = this.getCoordinate(b);
        return Math.abs(aa.x - bb.x) + Math.abs(aa.y - bb.y);
      }

    状态空间搜索

     searchA: function () {
        // debugger
        var _this = this;
        var n = 0;
        var lastState;
        var nowAllNode = [];
        var isSuccess = false;
        //估价函数
        var hx = _this.totalDis(_this.data.nowOrder) + _this.data.step;
        _this.data.Open.push({
          h: hx,
          state: _this.data.nowOrder
        });
        nowAllNode = nowAllNode.concat(_this.data.Open);
        var i = 0
        while (_this.data.Open.length !== 0) {
          i++
          var temp; //当前节点
          // var i = 0;
          temp = _this.deepCopy(_this.data.Open[0]);
          _this.data.Open.forEach(function (el, index) {
            if (el.h < temp.h) {
              temp = _this.deepCopy(el);
              i = index;
            }
            if (_this.data.Open.length == index + 1) {
              _this.data.Open.splice(i, 1);
            }
          })
          _this.data.Closed.push(temp);
          //是否为目标节点
          if (JSON.stringify(temp.state) == JSON.stringify(_this.data.purposeOder)) {
            wx.showToast({
              title: '完成了',
            })
            console.log("还差一步")
            isSuccess = true;
            break;
          }
          nowAllNode.push(temp);
          //防止陷入无限循环  
          if (nowAllNode.length > 10) {
            nowAllNode.splice(0, 1);
          }
          _this.data.Open = [];
          //扩展节点
          temp.state.forEach(function (el, index) {
            var nowA = _this.deepCopy(temp.state);
            if (_this.distance(el, temp.state[8]) == 1) {
              let s = el;
              nowA[index] = nowA[8];
              nowA[8] = s;
              var kz = {
                h: _this.data.step + _this.totalDis(nowA),
                state: nowA
              }
              var m = true;
              for (var i = 0; i < nowAllNode.length; i++) { //判断是否走过此节点
                if (JSON.stringify(kz.state) == JSON.stringify(nowAllNode[i].state)) m = false;
                if (nowAllNode.length == i + 1 && m) _this.data.Open.push(kz);
              }
            }
          })
          if (n != 0) {
            _this.data.step++;
            lastState = _this.data.nowOrder;
            _this.data.nowOrder = _this.data.Closed.pop().state;
            // 输出当前移动步骤结果
            // console.log('输出当前移动步骤结果:', JSON.stringify(_this.data.nowOrder));
            // _this.test();
          }
          n++;
        }
        if (!isSuccess) {
          wx.showToast({
            title: '没有找到',
          })
          console.log('没有找到');
        }
        return isSuccess

      },
     
    
    

    6.实现过程

    1.图片以六边形显示。

    2.打乱图片顺序。

    3.移动图片。

    4.找到图片最终拼成的步骤。

    7.动态效果

    8.代码地址

    github地址:git@e.coding.net:SpringSun/jigsaw/jigsaw.git 。

    出来混总是要还的
  • 相关阅读:
    Spring Boot 无侵入式 实现 API 接口统一 JSON 格式返回
    Java8 Stream:2万字20个实例,玩转集合的筛选、归约、分组、聚合
    排名前 16 的 Java 工具类
    万字详解,JDK8 的 Lambda、Stream 和日期的使用详解
    这样规范写代码,同事直呼“666”
    Java多线程高并发学习笔记——阻塞队列
    零散的MySQL基础总是记不住?看这一篇就够了!
    Java 中自定义注解及使用场景
    最简单的6种防止数据重复提交的方法!
    选择数据库联系人【选择之后将不在列表】
  • 原文地址:https://www.cnblogs.com/ping2yingshi/p/14483090.html
Copyright © 2020-2023  润新知