vue3项目 / 三子棋

参考: https://blog.csdn.net/iron_nini/article/details/130369453

App.vue

javascript 复制代码
<template>
  <div class="box">
    <div style="margin-bottom: 10px; font-size: 32px">{{ status }}</div>
    <div class="board-row">
      <squareComponent
        v-for="(item, index) in squareValues"
        :key="index"
        :text="item"
        @on-click="handleClick(index)"/>
    </div>
    <button id="again"
    @click="handleAgain"
    >Again</button>
  </div>
</template>

<script setup>
import { ref, watchEffect } from 'vue';
import squareComponent from './squareComponent';

const squareValues = ref(Array(9).fill(''));
const xIsNext = ref(true);
const status = ref('');


const calculateWinner = (squares) => {
  const lines = [
    [0, 1, 2],
    [3, 4, 5],
    [6, 7, 8],
    [0, 3, 6],
    [1, 4, 7],
    [2, 5, 8],
    [0, 4, 8],
    [2, 4, 6]
  ];

  for(let i = 0; i < lines.length; i++) {
    const [a, b, c] = lines[i];
    if(squares[a] && squares[a] === squares[b] && squares[a] === squares[c]) {
      return squares[a];
    }
  }

  return null;
}

watchEffect(() => {
  if(calculateWinner(squareValues.value)){
    status.value = 'Winner: ' + calculateWinner(squareValues.value) + ' ! ';
  }
  else if(!squareValues.value.filter(item => item === '').length){
    status.value = 'Draw !';
  }
  else{
    status.value = 'Next player is ' + (xIsNext.value ? 'X' : 'O');
  }
});

const handleClick = (index) => {
  if(squareValues[index] || calculateWinner(squareValues.value)) {
    return;
  }

  if(xIsNext.value) {
    squareValues.value[index] = 'X';
  }
  else {
    squareValues.value[index] = 'O';
  }

  xIsNext.value = !xIsNext.value;
}

const handleAgain = () => {
  squareValues.value = Array(9).fill('');
  xIsNext.value = true;
}
</script>

<style scoped>
.box {
  width: 100%;
  height: 100vh;
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: center;
  background: lightgreen;
  
}
.board-row {
    display: grid;
    grid-template-columns: repeat(3, minmax(0,1fr));
}
#again {
  font-size: 32px;
  font-family:'Times New Roman', Times, serif;
  margin-top: 20px;
}
</style>

squareComponent.vue

javascript 复制代码
<template>
  <button class="square" @click="handleClick">{{ text }}</button>
</template>

<script>
export default{
  name: 'squareComponent',
  props:{
    text:{
      type: String,
      default: ''
    }
  },
  emits: ['on-click'],

  setup(props, context){
    const handleClick = () =>{
      context.emit('on-click', props.text);
    };

    return {
      handleClick
    };
  },
};
</script>

<style scoped lang="less">
.square{
  box-sizing: border-box;
  width: 100px;
  height: 100px;
  margin: 10px;
  border-radius: 50%;
  background: pink;
  color: black;
  font-weight: 700;
  font-family: 'Times New Roman', Times, serif;
  font-size: 32px;
}
</style>

效果预览



相关推荐
有意义6 小时前
深度拆解分割等和子集:一维DP数组与倒序遍历的本质
前端·算法·面试
小怪点点6 小时前
vue3使用
前端·vue.js
Bigger7 小时前
CSS 这些年都经历了什么?一次看懂 CSS 的演化史
前端·css·前端工程化
DevUI团队7 小时前
🚀 【Angular】MateChat V20.2.2版本发布,新增8+组件,欢迎体验~
前端·javascript·人工智能
嚴寒7 小时前
前端配环境配到崩溃?这个一键脚手架让我少掉了一把头发
前端·react.js·架构
DevUI团队7 小时前
🚀 MateChat V1.11.0 震撼发布!新增工具按钮栏组件及体验问题修复,欢迎体验~
前端·javascript·人工智能
看晴天了7 小时前
新框架electronbun项目入门指南,解决electron体积大的难题,Electrobun:Electron 的轻量级革命 —— 12MB 应用 +
前端·架构
哇哇哇哇7 小时前
跨域:原因、解决方案CORS、JSONP、proxy、iframe(自用)
前端
Cache技术分享7 小时前
346. Java IO API - 操作文件和目录
前端·后端
滕青山7 小时前
HTTP状态查询 在线工具核心JS实现
前端·javascript·vue.js