生命不歇,挖坑不止!另一种挖洞算法的实现!

2023-10-22 16:40

本文主要是介绍生命不歇,挖坑不止!另一种挖洞算法的实现!,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这次就不用物理链条了,换一种方式实现。

回顾

在 物理挖洞-优化篇 和 物理挖洞-实现篇 中介绍了一种用多边形链条组件(cc.PhysicsChainCollider)实现物理挖洞的方法。这次打算用多边形碰撞组件(cc.PhysicsPolygonCollider)去实现物理挖洞。

建议先看前两篇的讲解,有助于更快理解这篇文章。

效果预览

微信小游戏-ios-端效果预览

实现步骤

整体思路是,先用 Clipper 去计算多边形,接着用 poly2tri 将多边形分割成多个三角形,最后用多边形刚体填充。

引入第三方库

Clipper

Clipper 是一个强大的用于多边形计算的运算库。前往下面这个地址下载,并作为插件导入 creator

http://jsclipper.sourceforge.net

为什么这次不用 物理挖洞-实现篇 中的 PolyBool 呢?

经测试发现 Clipper 的效率会比 PolyBool 高,并且 Clipper 内置了一个方法可以明确知道哪些多边形是洞。

poly2tri

poly2tri 是一个把多边形分割成三角形的库。下载地址如下:

https://github.com/r3mi/poly2tri.js

poly2tri 的使用有一些要注意的,大致就是不能有重复的点,不能有相交的形状。

初始化准备

先在场景中添加一个物理节点,一个绘图组件(用来画图)。

接着把物理引擎打开,监听触摸事件。

// onLoad() {
// 多点触控关闭
cc.macro.ENABLE_MULTI_TOUCH = false;
cc.director.getPhysicsManager().enabled = true;this.node_dirty.on(cc.Node.EventType.TOUCH_START, this._touchMove, this);
this.node_dirty.on(cc.Node.EventType.TOUCH_MOVE, this._touchMove, this);
// }

扩展多边形碰撞的组件

为了方便管理多边形碰撞组件,新建一个脚本 PhysicsPolygonColliderEx.ts

初始化

因为物理碰撞体需要物理刚体,我们可以加一些限制,并把这个菜单指向物理碰撞体的菜单中。

const { ccclass, property, menu, requireComponent } = cc._decorator;
@ccclass
@menu("i18n:MAIN_MENU.component.physics/Collider/PolygonEX-lamyoung.com")
@requireComponent(cc.RigidBody)
export default class PhysicsPolygonColliderEx extends cc.Component {
}

我们就可以在刚体节点中添加这个插件脚本了。

既然要用到多边形碰撞体,就定义一个多边形碰撞体数组。

private _physicsPolygonColliders: cc.PhysicsPolygonCollider[] = [];

因为 Clipper 中计算的结构是 {X,Y}

所以加个变量记录多边形顶点信息。

private _polys: { X: number, Y: number }[][] = [];

因为不同的库用的数据结构不同,所以添加两个转换方法。

private _convertVecArrayToClipperPath(poly: cc.Vec2[]) {return poly.map((p) => { return { X: p.x, Y: p.y } });
}private _convertClipperPathToPoly2triPoint(poly: { X: number, Y: number }[]) {return poly.map((p) => { return new poly2tri.Point(p.X, p.Y) });
}

加一个初始化数据的接口。

init(polys: cc.Vec2[][]) {this._polys = polys.map((v) => { return this._convertVecArrayToClipperPath(v) });
}

计算多边形

参考 Clipper 中的使用例子,写一个多边形差集调用。

//polyDifference(poly: cc.Vec2[]) {
const cpr = new ClipperLib.Clipper();
const subj_paths = this._polys;
const clip_paths = [this._convertVecArrayToClipperPath(poly)]
cpr.AddPaths(subj_paths, ClipperLib.PolyType.ptSubject, true);
cpr.AddPaths(clip_paths, ClipperLib.PolyType.ptClip, true);
const subject_fillType = ClipperLib.PolyFillType.pftEvenOdd;
const clip_fillType = ClipperLib.PolyFillType.pftEvenOdd;
const solution_polytree = new ClipperLib.PolyTree();
cpr.Execute(ClipperLib.ClipType.ctDifference, solution_polytree, subject_fillType, clip_fillType);
const solution_expolygons = ClipperLib.JS.PolyTreeToExPolygons(solution_polytree);
this._polys = ClipperLib.Clipper.PolyTreeToPaths(solution_polytree);

分割多边形并添加刚体

参考 poly2tri 中的使用,写一个多边形分割成三角形的调用。记得要把上面返回的数据转成 poly2tri 中可以使用的数据格式。

// polyDifference(poly: cc.Vec2[]) {
let _physicsPolygonColliders_count = 0;
for (const expolygon of solution_expolygons) {const countor = this._convertClipperPathToPoly2triPoint(expolygon.outer);const swctx = new poly2tri.SweepContext(countor);const holes = expolygon.holes.map(h => { return this._convertClipperPathToPoly2triPoint(h) });swctx.addHoles(holes);swctx.triangulate();const triangles = swctx.getTriangles();// 逐一处理三角形...
}

然后再逐一处理分割好的三角形,修改 cc.PhysicsPolygonColliderpoints 属性。

// 逐一处理三角形...
for (const tri of triangles) {let c = this._physicsPolygonColliders[_physicsPolygonColliders_count];if (!c) {//没有的话就创建c = this.addComponent(cc.PhysicsPolygonCollider);c.friction = 0;c.restitution = 0;this._physicsPolygonColliders[_physicsPolygonColliders_count] = c;}c.points = tri.getPoints().map((v, i) => {return cc.v2(v.x, v.y)});c.apply();_physicsPolygonColliders_count++;
}
// 剩余不要用的多边形清空。
this._physicsPolygonColliders.slice(_physicsPolygonColliders_count).forEach((v => {if (v.points.length) {v.points.length = 0;v.apply();}
}));

绘制泥土

只要在遍历三角形的时候逐点画线就行了。

if (i === 0) ctx.moveTo(v.x, v.y);
else ctx.lineTo(v.x, v.y);

添加命令队列

为了不让每帧计算量过多,添加一个命令队列。

private _commands: { name: string, params: any[] }[] = [];pushCommand(name: string, params: any[]) {this._commands.push({ name, params });
}

在每次更新的时候,取出几个命令去执行。

lateUpdate(dt: number) {if (this._commands.length) {// 每帧执行命令队列for (let index = 0; index < 2; index++) {const cmd = this._commands.shift();if (cmd)this[cmd.name](...cmd.params);elsebreak;}}
}

涂抹地形

整体思路和 物理挖洞-优化篇 和 物理挖洞-实现篇 差不多。不清楚的话,可以回看这两篇文章。

这次不同的是,加了一个涂抹步长控制,当涂抹间隔太小的时候,就不参与计算。

private _touchStartPos: cc.Vec2;
private _touchStart(touch: cc.Touch) {this._touchStartPos = undefined;this._touchMove(touch);
}private _touchMove(touch: cc.Touch) {const regions: cc.Vec2[] = [];const pos = this.graphics.node.convertToNodeSpaceAR(touch.getLocation());const count = DIG_FRAGMENT;if (!this._touchStartPos) {// 画一个圆(其实是多边形)for (let index = 0; index < count; index++) {const r = 2 * Math.PI * index / count;const x = pos.x + DIG_RADIUS * Math.cos(r);const y = pos.y + DIG_RADIUS * Math.sin(r);regions.push(this._optimizePoint([x, y]));}this._touchStartPos = pos;} else {const delta = pos.sub(this._touchStartPos);// 手指移动的距离太小的话忽略if (delta.lengthSqr() > 25) {// 这里是合并成一个顺滑的图形  详细上一篇文章const startPos = this._touchStartPos;for (let index = 0; index < count; index++) {const r = 2 * Math.PI * index / count;let vec_x = DIG_RADIUS * Math.cos(r);let vec_y = DIG_RADIUS * Math.sin(r);let x, y;if (delta.dot(cc.v2(vec_x, vec_y)) > 0) {x = pos.x + vec_x;y = pos.y + vec_y;} else {x = startPos.x + vec_x;y = startPos.y + vec_y;}regions.push(this._optimizePoint([x, y]));}this._touchStartPos = pos;}}if (regions.length)this.polyEx.pushCommand('polyDifference', [regions, this.graphics]);
}private _touchEnd(touch: cc.Touch) {this._touchStartPos = undefined;
}

小结

以上为白玉无冰使用 Cocos Creator v2.3.3 开发"物理挖洞之多边形碰撞体挖洞"的技术分享。如果对你有点帮助,欢迎分享给身边的朋友。

视频讲解

感谢各位看官的支持,如果这篇文章能给你带来一点点的帮助,白玉无冰感到非常愉快。如果可以点个在看的话,我会非常振奋开心。当然,请白玉无冰喝一杯奶茶的话,我会感到更加的满足。

其实,写这一篇图文花了一整天的时间(不算上之前的查阅资料及代码的实现),但是看到一篇文章的完成,富有满满的成就感。

不懂大家是否理解了这种实现方式,不知道是否需要补充一个视频讲解。

这样吧,如果这篇文章在看超过30,我就补录一篇视频讲解!

准备去研究新东西喽!冲呀!大家一定要多多实践练习!

点击“阅读原文”查看更多精彩

“在看”是最大的鼓励▼

这篇关于生命不歇,挖坑不止!另一种挖洞算法的实现!的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/262527

相关文章

Nginx实现高并发的项目实践

《Nginx实现高并发的项目实践》本文主要介绍了Nginx实现高并发的项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录使用最新稳定版本的Nginx合理配置工作进程(workers)配置工作进程连接数(worker_co

python中列表list切分的实现

《python中列表list切分的实现》列表是Python中最常用的数据结构之一,经常需要对列表进行切分操作,本文主要介绍了python中列表list切分的实现,文中通过示例代码介绍的非常详细,对大家... 目录一、列表切片的基本用法1.1 基本切片操作1.2 切片的负索引1.3 切片的省略二、列表切分的高

基于Python实现一个PDF特殊字体提取工具

《基于Python实现一个PDF特殊字体提取工具》在PDF文档处理场景中,我们常常需要针对特定格式的文本内容进行提取分析,本文介绍的PDF特殊字体提取器是一款基于Python开发的桌面应用程序感兴趣的... 目录一、应用背景与功能概述二、技术架构与核心组件2.1 技术选型2.2 系统架构三、核心功能实现解析

使用Python实现表格字段智能去重

《使用Python实现表格字段智能去重》在数据分析和处理过程中,数据清洗是一个至关重要的步骤,其中字段去重是一个常见且关键的任务,下面我们看看如何使用Python进行表格字段智能去重吧... 目录一、引言二、数据重复问题的常见场景与影响三、python在数据清洗中的优势四、基于Python的表格字段智能去重

Spring AI集成DeepSeek实现流式输出的操作方法

《SpringAI集成DeepSeek实现流式输出的操作方法》本文介绍了如何在SpringBoot中使用Sse(Server-SentEvents)技术实现流式输出,后端使用SpringMVC中的S... 目录一、后端代码二、前端代码三、运行项目小天有话说题外话参考资料前面一篇文章我们实现了《Spring

Nginx中location实现多条件匹配的方法详解

《Nginx中location实现多条件匹配的方法详解》在Nginx中,location指令用于匹配请求的URI,虽然location本身是基于单一匹配规则的,但可以通过多种方式实现多个条件的匹配逻辑... 目录1. 概述2. 实现多条件匹配的方式2.1 使用多个 location 块2.2 使用正则表达式

使用Apache POI在Java中实现Excel单元格的合并

《使用ApachePOI在Java中实现Excel单元格的合并》在日常工作中,Excel是一个不可或缺的工具,尤其是在处理大量数据时,本文将介绍如何使用ApachePOI库在Java中实现Excel... 目录工具类介绍工具类代码调用示例依赖配置总结在日常工作中,Excel 是一个不可或缺的工http://

SpringBoot实现导出复杂对象到Excel文件

《SpringBoot实现导出复杂对象到Excel文件》这篇文章主要为大家详细介绍了如何使用Hutool和EasyExcel两种方式来实现在SpringBoot项目中导出复杂对象到Excel文件,需要... 在Spring Boot项目中导出复杂对象到Excel文件,可以利用Hutool或EasyExcel

Python如何实现读取csv文件时忽略文件的编码格式

《Python如何实现读取csv文件时忽略文件的编码格式》我们再日常读取csv文件的时候经常会发现csv文件的格式有多种,所以这篇文章为大家介绍了Python如何实现读取csv文件时忽略文件的编码格式... 目录1、背景介绍2、库的安装3、核心代码4、完整代码1、背景介绍我们再日常读取csv文件的时候经常

Golang中map缩容的实现

《Golang中map缩容的实现》本文主要介绍了Go语言中map的扩缩容机制,包括grow和hashGrow方法的处理,具有一定的参考价值,感兴趣的可以了解一下... 目录基本分析带来的隐患为什么不支持缩容基本分析在 Go 底层源码 src/runtime/map.go 中,扩缩容的处理方法是 grow