基于React + Antd 实现的斗兽棋web应用

Posted 悠灵本林

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了基于React + Antd 实现的斗兽棋web应用相关的知识,希望对你有一定的参考价值。

项目简介

很高兴能以这种方式认识你,这个项目源自于学习了React官方的井字棋入门,想换一种方式,实战一下。

功能规划

  • 已实现

  • 未实现

    • 棋盘渲染
    • 棋子移动
    • 根据棋子特性的特别移动规则
    • 地形限制
    • 陷阱
    • 进入洞穴结束
    • 更改选棋
    • 贴图渲染
    • 界面优化

项目源码

具体源码前往:https://gitee.com/lin-liangyou/react-beast-fighting-chess

棋盘渲染

和React官方教学案例一样思路,将棋盘的棋格单独成组件,设立属性如mapitem,然后使用gamemap包裹起来处理。

mapitem

import React from 'react';
import './item.css';

const Item = (props)=>{
    return(
        <button 
        id={`type${props.typeId}`}
        onClick={props.onClick}
        className={`itemParent ${props.classType} ${props.isChoose}`}
        data-index={props.index}>
            {props.name}
        </button>
    )
}

export default Item;

gamemap(部分)

const Map = ()=>{
	//...
	const listRush = () =>{
    const mapAll = [];
    let mapLine = [];
    list.forEach(function(item,index) {
      const { listName, type } = mapInit(index);
      mapLine.push(
      <Item 
      key={index} 
      typeId={item}
      classType={(type === 'B')?'typeB':((type === 'A')?'typeA':'')}
      isChoose={ (choose[0] === index || choose[1] === index) ? (nextPlayer==='Red'?'typeACHO':'typeBCHO') : ''}
      name={listName||''} 
      onClick={ (event) => itemTouch(event) }
      index={index}/>);
      if( (index + 1) % 7 === 0 ){
        mapLine.push(<br/>);
        let parentLine = <div className="parentLine">{mapLine}</div>
        mapAll.push(parentLine);
        mapLine = [];
      }
    })
    return mapAll;
  }
    
    return (
      <div className='gameParent'>
        <div className='title'>{`next player:${nextPlayer}`}</div>
        {
          listRush()
        }
      </div>
    )
}
export default Map;

map.json

为了代码长度缩减,将用到的静态数据封装到map.json里面,list是地形布局,mapPOS则是每个棋盘对应的ID,其他的如此类推,ALsitBList则是存储棋子的数组,顺序按照listName

{
    "list": [
        0,0,2,3,2,0,0,
        0,0,0,2,0,0,0,
        0,0,0,0,0,0,0,
        0,1,1,0,1,1,0,
        0,1,1,0,1,1,0,
        0,1,1,0,1,1,0,
        0,0,0,0,0,0,0,
        0,0,0,2,0,0,0,
        0,0,2,3,2,0,0
    ],

    "mapPOS": [
         0,  1,  2,  3,  4,  5,  6,
         7,  8,  9, 10, 11, 12, 13,
        14, 15, 16, 17, 18, 19, 20,
        21, 22, 23, 24, 25, 26, 27,
        28, 29, 30, 31, 32, 33, 34,
        35, 36, 37, 38, 39, 40, 41,
        42, 43, 44, 45, 46, 47, 48,
        49, 50, 51, 52, 53, 54, 55,
        56, 57, 58, 59, 60, 61, 62
    ],

    "mapRiver":[
        22, 23, 25, 26,
        29, 30, 32, 33,
        36, 37, 39, 40
    ],

    "mapRiverBank":[
        15, 16, 18, 19,
        21, 24, 27,
        28, 31, 34,
        35, 38, 41,
        43, 44, 46, 47
    ],

    "mapTrap":[
        2, 4, 10, 52, 58 ,60
    ],

    "gameOver":[
        3, 59
    ],
    "listName":[
        "鼠","猫","狗","狼","豹","虎","狮","象"
    ],

    "AList":[
        20, 12,  8, 18, 16,  6,  0, 14
    ],

    "BList":[
        42, 50, 54, 44, 46, 56, 62, 48
    ]
}

棋子移动 && 规则特性 && 地形规则 && 游戏结束

gamemap(部分)

const itemTouch = (e)=>{
    let point = Number(e.target.dataset.index);
    //判断是否为首次选子
    if(choose[0] === 99){//选子状态
      //合理判断
      if(nextPlayer === 'Red'){
        if(AList.indexOf(point) === -1){
          message.warn('选子错误!');
          return;
        }else{//合理 -> 记录棋子类型
          setChessType( AList.indexOf(point) );
        }
      }else{
        if(BList.indexOf(point) === -1){
          message.warn('选子错误!');
          return;
        }else{//合理 -> 记录棋子类型
          setChessType( BList.indexOf(point) );
        }
      }
      //合理 -> 选子
      setChoose([point,99]);
    }else{//落子状态
      //落子逻辑 -> 落子点判断
      const fristPoint = mapPOS.indexOf(choose[0]);
      const comparePoint = mapPOS.indexOf(point);
      const number = comparePoint - fristPoint;
      if( Math.abs(number) === 7 || Math.abs(number) === 1){//距离计算 value:1
        //下水判断
        if( mapRiver.indexOf(point) !== -1 ){// 落子点 是否为水
          if(chessType === 0){

          }else{
            message.warn('落子错误,只有鼠才能入水!');
            return
          }
        }
      }else if(chessType === 6 || chessType === 5){
        console.log(`当前位置:${choose[0]}`);
        if(mapRiverBank.indexOf(choose[0]) !== -1){//河岸判断
          if( Math.abs(number) === 28 || Math.abs(number) === 3){//距离计算 value:3 || 2

          }else{
            message.warn('落子错误,你怎么跃的?!');
            return;
          }
        }else{
          message.warn('落子错误,前往河岸才可跳跃!');
          return;
        }
      }else{
        message.warn('落子错误,行动格错误!');
        return;
      }
      // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
      //   吃子 && 行动 逻辑过程
      // -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
      if(nextPlayer === 'Red'){//玩家判断 -> Red行动
        if( AList.indexOf(point) === -1){//踩子判断
          //吃子判断
          const enemy = BList.indexOf(point);
          if(enemy !== -1){// 落子点 踩敌判断
            // 踩敌 -> 大小比拼
            if( chessType >= enemy ){
              //列表移除
              BList[enemy] = 99;
            }else if( chessType === 0 && enemy === 7){//特殊点 1 鼠象关系
              //列表移除
              BList[enemy] = 99;
            }else if( mapTrap.indexOf(point) !== -1 ){//陷阱判断
              //列表移除
              BList[enemy] = 99;
            }else{
              message.warn(`落子错误,打不过!它比你大${enemy-chessType}`);
              return;
            }
          }
          //棋格行动
          const obj = AList.indexOf(choose[0]);
          AList[obj] = point; 
        }else{
          message.warn('落子错误,落子点有同伴!');
          return;
        }
      }else{//Blue行动
        if( BList.indexOf(point) === -1){//踩子判断
          //吃子判断
          const enemy = AList.indexOf(point);
          if(enemy !== -1){// 落子点 踩敌判断
            // 踩敌 -> 大小比拼
            if( chessType >= enemy ){
              //列表移除
              AList[enemy] = 99;
            }else if( chessType === 0 && enemy === 7){
              //列表移除
              AList[enemy] = 99;
            }else if( mapTrap.indexOf(point) !== -1 ){//陷阱判断
              //列表移除
              AList[enemy] = 99;
            }else{
              message.warn(`落子错误,打不过!它比你大${enemy-chessType}`);
              return;
            }
          }
          //棋格行动
          const obj = BList.indexOf(choose[0]);
          BList[obj] = point;
        }else{
          message.warn('落子错误,落子点有同伴!');
          return;
        }
      }
      //游戏结束状态判断
      if(gameOver.indexOf(point) !== -1){
        alert(`游戏结束,恭喜${nextPlayer}胜利\\n点击确认重新开始!!`);
        window.location.reload();
      }
      //落子还原 -=-=- 结束动作
      setChoose([99,99]);
      setChessType(null);
      if( nextPlayer === 'Red' )setNextPlayer('Blue');
      else setNextPlayer('Red');
    }
  }

新人第一次发帖,欢迎大家指正批评。

以上是关于基于React + Antd 实现的斗兽棋web应用的主要内容,如果未能解决你的问题,请参考以下文章

React 基于antd+video.js实现m3u8格式视频播放及实时切换

共享单车—— React后台管理系统开发手记:AntD Form基础组件

react + antd 实现打印功能(踩了不少坑)

共享单车—— React后台管理系统开发手记:AntD Table高级表格

共享单车—— React后台管理系统开发手记:AntD Table基础表格

放弃antd table,基于React手写一个虚拟滚动的表格