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),无需手动改

相关推荐
风酥糖2 小时前
Godot游戏练习01-第27节-升级选项选择生效
游戏·游戏引擎·godot
non-action_pilgrim5 小时前
《小坦克大战小怪兽》小游戏实战四:基于 protoactor-go 的游戏服务器框架与状态持久化实战
服务器·游戏·golang
apcipot_rain1 天前
Python实战——蒙特卡洛模拟分析杀牌游戏技能收益
python·游戏·数学建模
IpdataCloud1 天前
米哈游黑产案解析:游戏账号批量注册如何用IP查询识别外挂与多开用户?操作指南
网络协议·tcp/ip·游戏
私人珍藏库1 天前
【Android】GameNative 0.9.0 [特殊字符] 手机畅玩Steam游戏
android·游戏·智能手机·app·工具·软件·多功能
wanhengidc1 天前
云手机搬砖安全吗
大数据·运维·服务器·安全·游戏·智能手机
wanhengidc1 天前
服务器管理器的作用有哪些?
运维·服务器·网络·安全·游戏·智能手机
风酥糖1 天前
Godot游戏练习01-第26节-轮次结束后弹出升级选项
游戏·游戏引擎·godot
前端不太难1 天前
一天做出:鸿蒙 + AI 游戏 Demo
人工智能·游戏·harmonyos
世人万千丶2 天前
开源鸿蒙跨平台Flutter开发:成语接龙游戏应用
学习·flutter·游戏·华为·开源·harmonyos·鸿蒙