其他
Cocos Creator 物理刚体挖洞新方案(上)
The following article is from 白玉无冰 Author lamyoung
Hi,今天给大家分享一篇“物理刚体挖洞”教程,这次就不用理链条了,换一种方式实现。
回顾
在物理挖洞-优化篇和物理挖洞-实现篇中介绍了一种用多边形链条组件 (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.jspoly2tri
的使用有一些要注意的,大致就是不能有重复的点,不能有相交的形状。初始化准备
先在场景中添加一个物理节点,一个绘图组件(用来画图)。接着把物理引擎打开,监听触摸事件。
// 多点触控关闭
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
。初始化
因为物理碰撞体需要物理刚体,我们可以加一些限制,并把这个菜单指向物理碰撞体的菜单中。@ccclass
@menu("i18n:MAIN_MENU.component.physics/Collider/PolygonEX-lamyoung.com")
@requireComponent(cc.RigidBody)
export default class PhysicsPolygonColliderEx extends cc.Component {
}
我们就可以在刚体节点中添加这个插件脚本了。
既然要用到多边形碰撞体,就定义一个多边形碰撞体数组。
Clipper
中计算的结构是 {X,Y}
。所以加个变量记录多边形顶点信息。
因为不同的库用的数据结构不同,所以添加两个转换方法。
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) });
}
加一个初始化数据的接口。
this._polys = polys.map((v) => { return this._convertVecArrayToClipperPath(v) });
}
计算多边形
参考Clipper
中的使用例子,写一个多边形差集调用。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
中可以使用的数据格式。
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.PhysicsPolygonCollider
的 points
属性。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();
}
}));
绘制泥土
只要在遍历三角形的时候逐点画线就行了。else ctx.lineTo(v.x, v.y);
添加命令队列
为了不让每帧计算量过多,添加一个命令队列。pushCommand(name: string, params: any[]) {
this._commands.push({ name, params });
}
在每次更新的时候,取出几个命令去执行。
if (this._commands.length) {
// 每帧执行命令队列
for (let index = 0; index < 2; index++) {
const cmd = this._commands.shift();
if (cmd)
this[cmd.name](...cmd.params);
else
break;
}
}
}
涂抹地形
整体思路和物理挖洞-优化篇和物理挖洞-实现篇差不多。不清楚的话,可以回看这两篇文章。这次不同的是,加了一个涂抹步长控制,当涂抹间隔太小的时候,就不参与计算。
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
的技术分享,再次感谢白玉无冰童鞋,也欢迎关注[白玉无冰]获取完整项目哦~如果您在使用 Cocos Creator 2D/3D 的过程中 get 了独到的开发心得、见解或是方法,欢迎随时向我们投稿,帮助更多开发者们解决技术问题哦~
精彩预告:5 月 20 日下午 14 点 B 站直播间
《“大厂”的流量扶持策略》正式开麦
报名地址:http://hdxu.cn/uH2C8
包括华为在内的大厂有哪些开发者出海流量扶持策略?如何低成本获取用户,提高新增和留存?各个地区的发行新玩法?如何验证游戏与海外市场的契合度?
从开发到发行,面面俱到,欢迎报名喔!
最后我们的集赞兑礼包活动也要开奖啦~
恭喜@Wal-le@红领巾@江志锋@cmd@白玉无冰将获得我们的官方周边大礼包喔,也请五位童鞋看到消息后将联系方式及地址在后台留言告知我们喔~
更多精彩:腾讯光子团队是如何制作《最强魔斗士》的?520出海脱单指南,“大厂”的流量扶持策略Cocos Creator 物理挖洞教程!Creator 3D v1.1 里程碑版本,三大维度齐升级!腾讯光子《最强魔斗士》3D开发经验分享
新书推荐|零基础入门小游戏开发
你还在熬夜加班写 bug? 让小秘书来帮你
Cocos 插件开发者的福音来了,余额提现秒到账
Creator 2.3.3 更新说明,效率即是一切!
Creator 3D 官方中文视频教程,附素材源码
如何在 Creator 中优雅地嵌套 Prefab?
如何打通用户获取与变现的闭环实现稳定增长?
Analytics自定义事件功能详解,埋点分析利器Cocos Creator 开发原生游戏体验如何原生 3D 游戏《弹无虚发》是如何炼成的?微信小游戏首包超出4M之后技巧:微信如何设置星标??
前端开发者入门 Cocos Creator 必读