鸿蒙ArkTS+ArkUI实现五子棋游戏

打印 上一主题 下一主题

主题 926|帖子 926|积分 2780

鸿蒙ArkTS+ArkUI实现五子棋游戏

前言

  近期,鸿蒙系统热度飙升,引发了周围浩繁朋友的热烈探讨。出于这份浓厚的好奇心,我初步浏览了其官方文档,发现信息量庞大,全面消化需耗时很久并考验人的毅力。自踏入编程范畴以来,我不停秉持着“实践出真知”的原则,倾向于在动手操纵中意会文字背后的深意。恰逢我之前已利用Java和Python成功开辟过五子棋游戏,一个念头油然而生:何不借助鸿蒙系统,再次挑衅这一经典游戏的项目实践呢?正是这份灵感与刻意,催生了本篇文章的诞生。
  让我们一同利用鸿蒙系统,将五子棋游戏从Java和Python的实践迁移到鸿蒙平台,通过亲身实践深入明白鸿蒙的开辟魅力。这不但是一次技术尝试,更是一场布满乐趣与成长的学习之旅。
结果图




技术栈


  • 版本:HarmonyOS 5.0.3(15)
  • 编程语言:ArkTS
  • UI框架:ArkUI
源码

  这里只展示pages文件下的代码,如要获取整个项目,可以访问我的GitHub堆栈,不知道堆栈地址的朋友可以私聊我获取。

Index.ets —— 开始界面

  1. import { font, router } from '@kit.ArkUI';
  2. @Entry
  3. @Component
  4. struct Index {
  5.   // 注册字体
  6.   aboutToAppear() {
  7.     font.registerFont({
  8.       familyName: $r('app.string.index_title_font_name'),
  9.       familySrc: $r('app.string.index_title_font_src'),
  10.     })
  11.   }
  12.   build() {
  13.     Column() {
  14.       Row() {
  15.         Text($r("app.string.index_title"))
  16.           .fontSize($r('app.float.title_text_font_size'))
  17.           .fontWeight(FontWeight.Bold)
  18.           .fontFamily('SThuawenxingkai')
  19.       }
  20.       Row() {
  21.         Button() {
  22.           Text($r("app.string.start_button_text"))
  23.             .fontSize(20)
  24.             .fontWeight(FontWeight.Bold)
  25.             .fontColor('#000')
  26.             .fontFamily('SThuawenxingkai')
  27.         }
  28.         .onClick(() => {
  29.           router.pushUrl({ url: "pages/Second" })
  30.         })
  31.         .type(ButtonType.Capsule)
  32.         .startButtonStyle()
  33.       }
  34.     }
  35.     .height('100%')
  36.     .width('100%')
  37.     .justifyContent(FlexAlign.Center)
  38.     .backgroundImage($r("app.media.background_img"))
  39.     .backgroundImageSize(ImageSize.Cover)
  40.   }
  41.   @Styles
  42.   startButtonStyle() {
  43.     .margin({
  44.       top: 20
  45.     })
  46.     .backgroundColor($r('sys.color.button_background_color_transparent'))
  47.     .width('40%')
  48.     .height('6%')
  49.     .borderRadius(20)
  50.     .shadow({
  51.       radius: 10,
  52.       color: '#000',
  53.       offsetX: 0,
  54.       offsetY: 5
  55.     })
  56.   }
  57. }
复制代码
Second.ets —— 游戏界面

  1. // 坐标类型
  2. interface Coord {
  3.   x: number;
  4.   y: number;
  5.   color: boolean;
  6. }
  7. @Entry
  8. @Component
  9. struct Index {
  10.   private settings: RenderingContextSettings = new RenderingContextSettings(true)
  11.   private context: CanvasRenderingContext2D = new CanvasRenderingContext2D(this.settings)
  12.   // 棋盘起始坐标
  13.   board_x: number = 0
  14.   board_y: number = 0
  15.   // 棋盘宽度
  16.   board_width: number = 0
  17.   // 棋盘线的间距
  18.   line_spacing: number = 0
  19.   // true黑 false白
  20.   chess_color: boolean = true
  21.   // 棋子坐标
  22.   chess_coord: Coord[] = []
  23.   build() {
  24.     Flex({ direction: FlexDirection.Column, alignItems: ItemAlign.Center, justifyContent: FlexAlign.Center }) {
  25.       Canvas(this.context)
  26.         .width('100%')
  27.         .height('100%')
  28.         .backgroundColor($r("sys.color.ohos_id_color_background"))
  29.         .onReady(() => {
  30.           this.init()
  31.         })
  32.         .onClick((e) => {
  33.           this.drawChess(e.x, e.y)
  34.         })
  35.         .backgroundImage($r("app.media.background_img"))
  36.         .backgroundImageSize(ImageSize.Cover)
  37.     }
  38.     .width('100%')
  39.     .height('100%')
  40.   }
  41.   // 初始化棋盘方法
  42.   init() {
  43.     // 计算宽度,适配横屏
  44.     if (this.context.width < this.context.height) {
  45.       this.board_width = this.context.width - 40
  46.     } else {
  47.       this.board_width = this.context.height - 40
  48.     }
  49.     this.board_x = (this.context.width - this.board_width) / 2
  50.     this.board_y = (this.context.height - this.board_width) / 2
  51.     // 绘制正方形
  52.     this.context.lineWidth = 4
  53.     this.context.fillStyle = '#DBC26F'; // 设置填充颜色
  54.     this.context.strokeStyle = '#000'
  55.     this.context.fillRect(this.board_x, this.board_y, this.board_width, this.board_width); // 填充圆形
  56.     this.context.strokeRect(this.board_x, this.board_y, this.board_width, this.board_width)
  57.     // 计算棋盘线间距
  58.     this.line_spacing = this.board_width / 14
  59.     this.context.beginPath(); // 开始新路径
  60.     this.context.lineWidth = 2
  61.     for (let i = 0; i < 14; i++) {
  62.       this.context.moveTo(this.board_x, this.board_y + i * this.line_spacing)
  63.       this.context.lineTo(this.board_x + this.board_width, this.board_y + i * this.line_spacing)
  64.     }
  65.     for (let i = 0; i < 14; i++) {
  66.       this.context.moveTo(this.board_x + i * this.line_spacing, this.board_y)
  67.       this.context.lineTo(this.board_x + i * this.line_spacing, this.board_y + this.board_width)
  68.     }
  69.     this.context.stroke() // 绘制
  70.     // 绘制棋盘上的五个点
  71.     const points: Coord[] = [
  72.       { x: 3, y: 3, color: true },
  73.       { x: 11, y: 3, color: true },
  74.       { x: 3, y: 11, color: true },
  75.       { x: 11, y: 11, color: true },
  76.       { x: 7, y: 7, color: true }
  77.     ];
  78.     this.context.fillStyle = '#000'; // 设置填充颜色
  79.     this.context.lineWidth = 1; // 设置描边宽度
  80.     for (const point of points) {
  81.       this.context.beginPath(); // 新路径
  82.       const x = this.board_x + this.line_spacing * point.x;
  83.       const y = this.board_y + this.line_spacing * point.y;
  84.       this.context.arc(x, y, 4, 0, 2 * Math.PI); // 绘制圆形路径
  85.       this.context.fill(); // 填充圆形
  86.     }
  87.   }
  88.   // 绘制棋子
  89.   drawChess(x: number, y: number) {
  90.     // 判断点击区域是否在棋盘内
  91.     if (x >= this.board_x && x <= this.board_x + this.board_width && y >= this.board_y &&
  92.       y <= this.board_y + this.board_width) {
  93.       // 判断并设置棋子颜色
  94.       if (this.chess_color) {
  95.         this.context.fillStyle = '#000';
  96.       } else {
  97.         this.context.fillStyle = '#FFF';
  98.       }
  99.       // 修正坐标
  100.       x = Math.round((x - this.board_x) / this.line_spacing);
  101.       y = Math.round((y - this.board_y) / this.line_spacing);
  102.       // 判断是否已经有棋子
  103.       for (const coord of this.chess_coord) {
  104.         if (coord.x === x && coord.y === y) {
  105.           return
  106.         }
  107.       }
  108.       // 绘制棋子
  109.       this.context.strokeStyle = '#000'; // 设置描边颜色
  110.       this.context.beginPath(); // 新路径
  111.       this.context.arc(x * this.line_spacing + this.board_x, y * this.line_spacing + this.board_y,
  112.         this.line_spacing / 2 - 1, 0, 2 * Math.PI); // 绘制圆形路径
  113.       this.context.fill(); // 填充圆形
  114.       this.context.stroke(); // 描边圆形
  115.       // 加入棋子坐标
  116.       this.chess_coord.push({ x, y, color: this.chess_color })
  117.       // 判断是否五子连珠
  118.       this.checkLink()
  119.       // 切换棋子颜色
  120.       this.chess_color = !this.chess_color
  121.     }
  122.   }
  123.   // 判断是否五子连珠
  124.   checkLink() {
  125.     if (!this.chess_coord || this.chess_coord.length === 0) {
  126.       return;
  127.     }
  128.     const lastCoord = this.chess_coord[this.chess_coord.length - 1];
  129.     const directions: [number, number][] = [
  130.       [1, 0], // 垂直方向
  131.       [0, 1], // 水平方向
  132.       [1, 1], // 右对角线
  133.       [-1, 1]// 左对角线
  134.     ];
  135.     for (const direction of directions) {
  136.       for (let method = 0; method < 5; method++) {
  137.         const linkingCoords: Coord[] = [];
  138.         for (let i = 0; i < 5; i++) {
  139.           const newCoord: Coord = {
  140.             x: lastCoord.x + direction[0] * (i - method),
  141.             y: lastCoord.y + direction[1] * (i - method),
  142.             color: lastCoord.color
  143.           };
  144.           const found = this.chess_coord.find(coord =>
  145.           coord.x === newCoord.x && coord.y === newCoord.y && coord.color === newCoord.color
  146.           );
  147.           if (found) {
  148.             linkingCoords.push(found)
  149.           }
  150.         }
  151.         if (linkingCoords.length >= 5) {
  152.           linkingCoords.forEach(item => {
  153.             console.log(item.x.toString(), item.y.toString(), item.color)
  154.             this.context.strokeStyle = '#FEFE00'; // 设置描边颜色
  155.             this.context.beginPath(); // 新路径
  156.             this.context.arc(item.x * this.line_spacing + this.board_x, item.y * this.line_spacing + this.board_y,
  157.               this.line_spacing / 2 - 1, 0, 2 * Math.PI); // 绘制圆形路径
  158.             this.context.stroke(); // 描边圆形
  159.           })
  160.           AlertDialog.show({
  161.             message: `恭喜${this.chess_color ? '黑棋' : '白棋'},你赢了!`,
  162.             cancel: () => {
  163.               // 初始化棋盘
  164.               this.chess_coord = []
  165.               this.chess_color = true
  166.               // 初始化棋盘
  167.               this.context.clearRect(0, 0, this.context.width, this.context.height);
  168.               this.init()
  169.             }
  170.           });
  171.         }
  172.       }
  173.     }
  174.   }
  175. }
复制代码
竣事语

  经过这一系列的探索与实践,我们不但成功地将五子棋游戏实现到了鸿蒙系统之上,更在这一过程中深刻体会到了鸿蒙平台的强盛功能与无穷潜力。从初步打仗鸿蒙的渺茫,到逐步熟悉并掌握其开辟技巧,每一次挑衅都见证了我们的成长与进步。如今,回首这段旅程,我们劳绩的不但仅是技术上的提升,更有对编程热爱的加深,以及对未来无穷可能的向往。愿这份经历可以或许鼓励更多开辟者,勇敢地在鸿蒙的世界里探索、实践、创新,共同书写属于我们的精彩篇章。
  由于我也是初学者,现在实现的功能非常简朴,如果后续有功能的更新,将实时更新到GitHub堆栈。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

拉不拉稀肚拉稀

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表