”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > HTML 和 CSS 中的“拼图”滑块游戏

HTML 和 CSS 中的“拼图”滑块游戏

发布于2024-08-15
浏览:632

\

跳至拼图

有一天,我的脑海中突然浮现出童年时的一个小拼图玩具的回忆,这是一个滑块拼图,其中 15 个方形瓷砖以 4 x 4 的单元格排列方式放置在框架中,留下一个可用空间。每块瓷砖和框架边缘上的一组脊和凹槽允许瓷砖彼此滑过,同时将瓷砖固定在框架中。在任何给定时间,与自由空间相邻的任何图块都可以移动到该空间中,否则这些图块将被阻止移动。将一块图块移动到空闲空间中,然后在该图块来自的地方留下一个新的空闲空间,然后另一个图块可以移动到该新空间中。这个想法是通过以这种方式重复滑动图块来将图块排列成某种预定的顺序。

显然这被称为“15 Puzzle”,自 1870 年代以来就一直存在。搜索网络会返回许多用各种编程语言编写的游戏,实际上 dev.to 上有几篇文章,包括 https://dev.to/artydev/let-us-code-a-sliding-puzzle-9n 、https://dev.to/xzanderzone/making-a-slider-puzzle-in-java-script-83m 和 https://dev.to/claurcia/slide-puzzle-5c55 均采用 JavaScript 和 https://dev.to/claurcia/slide-puzzle-5c55 Go 中的 /dev.to/mfbmina/building-a-sliding-puzzle-with-go-3bnj。对于那些学习 JavaScript 的人来说,它也是一个很好的入门挑战。

不过,激起我兴趣的是它应该可以在网络上使用根本不使用编程语言来重新创建的想法!也就是说,仅使用纯 HTML 和 CSS 的实现。所以我在下面介绍一下。我必须做出的一个妥协是,所提供的 10 场比赛具有固定的预先洗牌的起始位置。

为此,预定的顺序是显示完整的图片。

此实现的基本原理是每个图块保留其在帧内位置的状态记录。在 HTML 和 CSS 中更改和保持状态的方法并不多,但最常见的是“复选框黑客”,并且此实现大量使用了它。对于任何不熟悉复选框黑客的人来说,当单击或点击

因此,每个图块都有一对单选按钮组,每个单选按钮组有四个单选按钮。其中一组保留图块在 X 轴上的位置,另一组保留图块在 Y 轴上的位置。 (如果您愿意,也可以分别是水平位置和垂直位置。)这 15 个图块最初通过其单选按钮指定了不同的 X 和 Y 坐标组合,以便每个图块占据帧中的不同单元格。
 
这些图块最初放置在框架的顶部、左侧单元格中,然后通过 CSS 在框架内移动,通过对单选按钮应用平移变换来测量单选按钮的状态:

/* "X" refers to the X-axis cell positions, "Y" to the Y-axis cell positions. 
 * 0, 1, 2, 3 refers to the position on that axis, 
 * 0 = left and top, 3 = right and bottom respectively.
 */

.tile:has(.X0:checked~.Y0:checked) {
     transform: translate(0%, 0%);
}
.tile:has(.X0:checked~.Y1:checked) {
     transform: translate(0%, 100%);
}
.tile:has(.X0:checked~.Y2:checked) {
     transform: translate(0%, 200%);
}
.tile:has(.X0:checked~.Y3:checked) {
     transform: translate(0%, 300%);
}
.tile:has(.X1:checked~.Y0:checked) {
     transform: translate(100%, 0%);
}
/* and so on for the remainder of the sixteen combinations */

该图块还包含八个标签元素,对应于八个单选按钮。每个标签都是绝对定位的,彼此重叠,并且每个标签都完全填充图块。标签是透明的,并且最初设置为不响应单击和点击,方法是在所有标签上设置pointer-events:none。

下一步是 CSS 选择器识别空单元格的位置。这是通过消除来完成的,它是 X、Y 坐标不由 15 个图块中任何一个的单选按钮组对表示的单元格。

例如,如果匹配:

.frame:not(:has(.tile .X0:checked~.Y0:checked)) { .... }

那么空单元格当前必须位于左上角单元格中。对 16 个单元格中的每一个单元格重复此操作,其中恰好有一个单元格将匹配。

一旦完成,就可以识别与空单元格相邻的单元格。如果空单元格位于角落,则正好有两个图块可以移动到该单元格中,否则,如果空单元格靠着框架的一侧,则有三个图块可以移动到该单元格中,否则空单元格可以移动到该单元格中。单元格必须是四个中间单元格之一,并且有四个方块可以移动到它。对于每个图块,图块的八个标签之一将激活将图块移动到空单元格所需的正确单选按钮。通过将其pointer-events 值设置回auto 来启用该标签。举个例子:

/* Top, left corner */
.frame:not(:has(.tile .X0:checked ~ .Y0:checked)) {
  :is(
      .tile:has(.X0:checked ~ .Y1:checked) label.Y0,
      .tile:has(.X1:checked ~ .Y0:checked) label.X0
    ) {
    pointer-events: auto;
  }
}

/* right most cell of row two */
.frame:not(:has(.tile .X1:checked ~ .Y3:checked)) {
  :is(
      .tile:has(.X1:checked ~ .Y2:checked) label.Y3,
      .tile:has(.X0:checked ~ .Y3:checked) label.X1,
      .tile:has(.X2:checked ~ .Y3:checked) label.X1
    ) {
    pointer-events: auto;
  }
}

/* second cell from left on row three */
.frame:not(:has(.tile .X2:checked ~ .Y1:checked)) {
  :is(
      .tile:has(.X2:checked ~ .Y0:checked) label.Y1,
      .tile:has(.X2:checked ~ .Y2:checked) label.Y1,
      .tile:has(.X1:checked ~ .Y1:checked) label.X2,
      .tile:has(.X3:checked ~ .Y1:checked) label.X2
    ) {
    pointer-events: auto;
  }
}

游戏的最后一步是确定谜题何时被解决。这只是检查 15 个图块是否都将其预期的 X 和 Y 轴单选按钮设置为“已解决”位置的情况。

/* Each tile is assigned a letter "a" to "o". 
 * The puzzle is solved when the tiles are in alphabetical order
 * reading left to right and top to bottom
*/
.frame:has(.a .X0:checked ~ .Y0:checked):has(.b .X1:checked ~ .Y0:checked):has(
    .c .X2:checked ~ .Y0:checked
  ):has(.d .X3:checked ~ .Y0:checked):has(.e .X0:checked ~ .Y1:checked):has(
    .f .X1:checked ~ .Y1:checked
  ):has(.g .X2:checked ~ .Y1:checked):has(.h .X3:checked ~ .Y1:checked):has(
    .i .X0:checked ~ .Y2:checked
  ):has(.j .X1:checked ~ .Y2:checked):has(.k .X2:checked ~ .Y2:checked):has(
    .l .X3:checked ~ .Y2:checked
  ):has(.m .X0:checked ~ .Y3:checked):has(.n .X1:checked ~ .Y3:checked):has(
    .o .X2:checked ~ .Y3:checked
  )
  ~ .options
  .success {
  display: block;
}

剩下的都是装饰性的。滑动是通过上述变换的简单转换来完成的

.tile {
  transition: 0.5s transform;
  @media (prefers-reduced-motion) {
    transition: none;
  }
}

每个图块使用背景大小和背景位置显示游戏图像的一部分

.tile {
  background-size: 400%;
}
#board1 .tile {
  background-image: url("https://alohci.net/image/dev.to/slidergame/mullermarc-k7bQqdUf954-unsplash.webp");
}
.a {
  background-position: 0% 0%;
}
.b {
  background-position: 33.333% 0%;
}
.c {
  background-position: 66.667% 0%;
}
.d {
  background-position: 100% 0%;
}
.e {
  background-position: 0% 33.333%;
}
/* and so on for the remaining tiles */

还有一组单选按钮可以选择玩十个游戏中的哪一个。

要玩游戏,只需点击或点按您想要滑动到空单元格的图块即可。

我还提供了“准系统”模式来显示平铺字母和单选按钮,这可能有助于理解 HTML 和 CSS 的工作原理。

这就是完成的益智游戏。如果您有任何反馈,请告诉我。


版本声明 本文转载于:https://dev.to/alohci/15-puzzle-slider-game-in-html-and-css-5g2i?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 如何使用 Laravel Eloquent 的 firstOrNew() 方法有效优化 CRUD 操作?
    如何使用 Laravel Eloquent 的 firstOrNew() 方法有效优化 CRUD 操作?
    使用 Laravel Eloquent 优化 CRUD 操作在 Laravel 中使用数据库时,插入或更新记录是很常见的。为了实现这一点,开发人员经常求助于条件语句,在决定执行插入或更新之前检查记录是否存在。firstOrNew() 方法幸运的是, Eloquent 通过firstOrNew() 方...
    编程 发布于2024-11-06
  • 为什么在 PHP 中重写方法参数违反了严格的标准?
    为什么在 PHP 中重写方法参数违反了严格的标准?
    在 PHP 中重写方法参数:违反严格标准在面向对象编程中,里氏替换原则 (LSP) 规定:子类型的对象可以替换其父对象,而不改变程序的行为。然而,在 PHP 中,用不同的参数签名覆盖方法被认为是违反严格标准的。为什么这是违规?PHP 是弱类型语言,这意味着编译器无法在编译时确定变量的确切类型。这意味...
    编程 发布于2024-11-06
  • 哪个 PHP 库提供卓越的 SQL 注入防护:PDO 还是 mysql_real_escape_string?
    哪个 PHP 库提供卓越的 SQL 注入防护:PDO 还是 mysql_real_escape_string?
    PDO vs. mysql_real_escape_string:综合指南查询转义对于防止 SQL 注入至关重要。虽然 mysql_real_escape_string 提供了转义查询的基本方法,但 PDO 成为了一种具有众多优点的卓越解决方案。什么是 PDO?PHP 数据对象 (PDO) 是一个数...
    编程 发布于2024-11-06
  • React 入门:初学者的路线图
    React 入门:初学者的路线图
    大家好! ? 我刚刚开始学习 React.js 的旅程。这是一次令人兴奋(有时甚至具有挑战性!)的冒险,我想分享一下帮助我开始的步骤,以防您也开始研究 React。这是我的处理方法: 1.掌握 JavaScript 基础知识 在开始使用 React 之前,我确保温习一下我的 JavaScript 技...
    编程 发布于2024-11-06
  • 如何引用 JavaScript 对象中的内部值?
    如何引用 JavaScript 对象中的内部值?
    如何在 JavaScript 对象中引用内部值在 JavaScript 中,访问引用同一对象中其他值的对象中的值有时可能具有挑战性。考虑以下代码片段:var obj = { key1: "it ", key2: key1 " works!" }; ...
    编程 发布于2024-11-06
  • Python 列表方法快速指南及示例
    Python 列表方法快速指南及示例
    介绍 Python 列表用途广泛,并附带各种内置方法,有助于有效地操作和处理数据。下面是所有主要列表方法的快速参考以及简短的示例。 1. 追加(项目) 将项目添加到列表末尾。 lst = [1, 2, 3] lst.append(4) # [1, 2, 3, 4]...
    编程 发布于2024-11-06
  • C++ 中何时需要用户定义的复制构造函数?
    C++ 中何时需要用户定义的复制构造函数?
    何时需要用户定义的复制构造函数?复制构造函数是 C 面向对象编程的组成部分,提供了一种基于现有实例初始化对象的方法。虽然编译器通常会为类生成默认的复制构造函数,但在某些情况下需要进行自定义。需要用户定义复制构造函数的情况当默认复制构造函数不够时,程序员会选择用户定义的复制构造函数来实现自定义复制行为...
    编程 发布于2024-11-06
  • 尝试...捕获 V/s 安全分配 (?=):现代发展的福音还是诅咒?
    尝试...捕获 V/s 安全分配 (?=):现代发展的福音还是诅咒?
    最近,我发现了 JavaScript 中引入的新安全赋值运算符 (?.=),我对它的简单性着迷。 ? 安全赋值运算符 (SAO) 是传统 try...catch 块的简写替代方案。它允许您内联捕获错误,而无需为每个操作编写显式的错误处理代码。这是一个例子: const [error, respons...
    编程 发布于2024-11-06
  • 如何在Python中优化固定宽度文件解析?
    如何在Python中优化固定宽度文件解析?
    优化固定宽度文件解析为了有效地解析固定宽度文件,可以考虑利用Python的struct模块。此方法利用 C 来提高速度,如以下示例所示:import struct fieldwidths = (2, -10, 24) fmtstring = ' '.join('{}{}'.format(abs(fw...
    编程 发布于2024-11-06
  • 蝇量级
    蝇量级
    结构模式之一旨在通过与相似对象共享尽可能多的数据来减少内存使用。 在处理大量相似对象时特别有用,为每个对象创建一个新实例在内存消耗方面会非常昂贵。 关键概念: 内在状态:多个对象之间共享的状态独立于上下文,并且在不同对象之间保持相同。 外部状态:每个对象唯一的、从客户端传递的状态。此状态可能会有所不...
    编程 发布于2024-11-06
  • 解锁您的 MySQL 掌握:MySQL 实践实验室课程
    解锁您的 MySQL 掌握:MySQL 实践实验室课程
    通过全面的 MySQL 实践实验室课程提高您的 MySQL 技能并成为数据库专家。这种实践学习体验旨在指导您完成一系列实践练习,使您能够克服复杂的 SQL 挑战并优化数据库性能。 深入了解 MySQL 无论您是想要建立强大 MySQL 基础的初学者,还是想要提升专业知识的经验丰富的开...
    编程 发布于2024-11-06
  • 文件夹
    文件夹
    ? ?大家好,我是尼克?? 利用专家工程解决方案提升您的项目 探索我的产品组合,了解我如何将尖端技术、强大的问题解决能力和创新热情结合起来,构建可扩展的高性能应用程序。无论您是寻求增强开发流程还是解决复杂的技术挑战,我都可以帮助您实现愿景。看看我的工作,让我们合作做一些非凡的事情! 在这里联系我:作...
    编程 发布于2024-11-06
  • 通过 Gmail 发送电子邮件时如何修复“SMTP Connect() 失败”错误?
    通过 Gmail 发送电子邮件时如何修复“SMTP Connect() 失败”错误?
    SMTP 连接失败:解决“SMTP Connect() 失败”错误尝试使用 Gmail 发送电子邮件时,您可能会遇到错误消息指出“SMTP -> 错误:无法连接到服务器:连接超时 (110)\nSMTP Connect() 失败。消息未发送。\n邮件程序错误:SMTP Connect() 失败。”此...
    编程 发布于2024-11-06
  • 如何使用 Pillow 在 Python 中水平连接多个图像?
    如何使用 Pillow 在 Python 中水平连接多个图像?
    用Python水平连接图像水平组合多个图像是图像处理中的常见任务。 Python 提供了强大的工具来使用 Pillow 库来实现此目的。问题描述考虑三个尺寸为 148 x 95 的方形 JPEG 图像。目标是水平连接这些图像图像,同时避免结果输出中出现任何部分图像。建议的解决方案以下代码片段解决了该...
    编程 发布于2024-11-06

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3