Cocos Creator3.8.0 Tiled地图三合一完整脚本(加载+兼容性校验+坐标互转,一键可用,适配Tiled1.4.x)

以下是实用型代码,仅供参考。

import { _decorator, Component, Node, TiledMap, Vec2, Input, input, EventTouch, resources, error, log } from 'cc';

const { ccclass, property } = _decorator;

@ccclass('TiledMapFullTool')

export class TiledMapFullTool extends Component {

@property({tooltip: "resources下地图路径,无.tmx后缀(例:map/gameMap)"})

mapPath: string = "";

@property({tooltip: "地图节点名称"})

mapNodeName: string = "GameTiledMap";

private tiledMap: TiledMap = null; // 地图组件缓存

start() {

// 一键执行:加载→校验→绑定坐标功能

this.loadAndInitMap();

}

// 1. 地图加载核心(动态创建+挂载)

async loadAndInitMap() {

if(!this.mapPath) {error("请填写resources下地图路径!"); return;}

log("开始加载Tiled地图...");

try {

// 动态创建地图节点+组件

let mapNode = this.node.getChildByName(this.mapNodeName);

if(!mapNode) {

mapNode = new Node(this.mapNodeName);

this.node.addChild(mapNode);

}

mapNode.setAnchorPoint(0, 0); // 必设,对齐Tiled原点

this.tiledMap = mapNode.getComponent(TiledMap) || mapNode.addComponent(TiledMap);

// 加载地图资源

const tmxRes = await resources.load<TiledMap>(this.mapPath);

this.tiledMap.tmxAsset = tmxRes;

log("✅ 地图加载成功");

// 加载后自动校验+绑定坐标功能

this.checkMapCompatibility();

this.bindClickGetTile();

} catch (e) {

error(`❌ 地图加载失败:${e}`);

log("💡 排查:路径错误/资源不在resources/格式非XML");

}

}

// 2. 兼容性自动校验(4大核心项)

checkMapCompatibility() {

if(!this.tiledMap) return;

log("开始地图兼容性校验...");

// 校验1:地图格式(必须XML)

if(this.tiledMap.tmxAsset?.rawXml) log("✅ 地图格式为XML,符合要求");

else error("❌ 地图格式错误!Tiled需导出XML格式,禁用二进制/CSV");

// 校验2:图层(无中文/空格)

const invalidLayers = this.tiledMap.getLayers().filter(l => l.name.includes(" ") || /[\u4e00-\u9fa5]/.test(l.name));

invalidLayers.length > 0 ? error(`❌ 非法图层名:${invalidLayers.map(l=>l.name)}`) : log("✅ 图层校验通过");

// 校验3:瓦片集(无缺失)

this.tiledMap.getTileSets().length > 0 ? log("✅ 瓦片集加载正常") : error("❌ 瓦片集缺失!tmx/tsx/png需同目录");

// 校验4:路径(无中文)

/[\u4e00-\u9fa5]/.test(this.mapPath) ? error("❌ 地图路径含中文") : log("✅ 地图路径合法");

this.tiledMap.getLayers().length>0 ? log("🎉 所有校验通过,地图可用") : error("❌ 地图无有效图层");

}

// 3. 坐标互转核心(世界↔瓦片)

// 世界坐标转瓦片坐标

worldToTile(worldPos: Vec2): Vec2 {

if(!this.tiledMap) return Vec2.ZERO;

const tileSize = this.tiledMap.tileSize;

const mapWorldPos = this.tiledMap.node.worldPosition;

const x = Math.floor((worldPos.x - mapWorldPos.x) / tileSize.width);

const y = Math.floor((mapWorldPos.y - worldPos.y) / tileSize.height);

return new Vec2(x, y);

}

// 瓦片坐标转世界坐标(默认返回瓦片中心)

tileToWorld(tilePos: Vec2): Vec2 {

if(!this.tiledMap) return Vec2.ZERO;

const tileSize = this.tiledMap.tileSize;

const mapWorldPos = this.tiledMap.node.worldPosition;

const worldX = mapWorldPos.x + tilePos.x * tileSize.width + tileSize.width/2;

const worldY = mapWorldPos.y - tilePos.y * tileSize.height - tileSize.height/2;

return new Vec2(worldX, worldY);

}

// 实用功能:点击地图打印瓦片坐标

bindClickGetTile() {

input.on(Input.EventTouch.END, (event: EventTouch) => {

if(!this.tiledMap) return;

const worldPos = event.getUILocation();

const tilePos = this.worldToTile(worldPos);

log("当前点击瓦片坐标:X="+tilePos.x+",Y="+tilePos.y);

}, this);

}

}

✅ 3步零踩坑使用(直接上线)

  1. 脚本挂载到Canvas节点,无需额外节点

  2. 属性面板填2个参数:①mapPath(resources下地图路径,无后缀)②地图节点名(默认即可)

  3. 运行项目,自动完成加载→校验→坐标可用,控制台看结果

🚨 终极避坑(必看)

  1. 地图资源:tmx+tsx+瓦片png 必须同目录,且全放resources文件夹

  2. Tiled导出:XML格式+外部tsx+兼容1.4格式,禁用高版本特性

  3. 坐标系:地图锚点已自动适配(0,0),无需手动改

相关推荐
不穿格子的程序员8 小时前
从零开始刷算法——贪心篇1:跳跃游戏1 + 跳跃游戏2
算法·游戏·贪心
微祎_9 小时前
Flutter for OpenHarmony:构建一个 Flutter 镜像绘图游戏,对称性认知、空间推理与生成式交互设计
flutter·游戏·交互
前端不太难10 小时前
HarmonyOS 游戏项目,从 Demo 到可上线要跨过哪些坑
游戏·状态模式·harmonyos
子春一10 小时前
Flutter for OpenHarmony:色彩捕手:基于 CIELAB 色差模型与人眼感知的高保真色彩匹配游戏架构解析
flutter·游戏·架构
前端不太难14 小时前
在 HarmonyOS 上,游戏状态该怎么“死而复生”
游戏·状态模式·harmonyos
ujainu1 天前
Flutter + OpenHarmony 游戏开发进阶:用户输入响应——GestureDetector 实现点击发射
flutter·游戏·openharmony
ujainu1 天前
Flutter + OpenHarmony 实现无限跑酷游戏开发实战—— 对象池化、性能优化与流畅控制
flutter·游戏·性能优化·openharmony·endless runner
呆呆敲代码的小Y1 天前
【Unity工具篇】| 超实用工具LuBan,快速上手使用
游戏·unity·游戏引擎·unity插件·luban·免费游戏·游戏配置表
我的offer在哪里1 天前
用 Unity 从 0 做一个「可以玩的」游戏,需要哪些步骤和流程
游戏·unity·游戏引擎
串流游戏联盟1 天前
启程!手机也能邂逅暖暖万相奇观
游戏·远程工作