ITU(Isometric Tiled 2 Unity) is the tool for analyze Tiled map and export it to Unity.
it only focus on isometric base tile.
i'm not plan to wrote a All-In-One tool inside of that , i try to make ITU process flow more separately more independently.
so that it's can be more easily to modify or use each part with extension via script.
- add : support for Tilest "Collection of image" type (not export anything , need use extension to handle the custom property)
- optimize : not generate tileset material if it never been used.
- init version
via e-mail: iamzealotwang@126.com
language : 中文 or English
-
Isometric , Isometric Staggered , Infinite Map support
-
Custom properties support.
-
Custom Shader support.
-
Custom render chunk size support , extremely useful for huge map.
-
Custom tile size (not 2:1 eg 256x128,128x64) support
-
Low level API , easy to merge with existing workflow.
-
Full source code access and easy to modify.
-
ITU(Isometric Tiled 2 Unity) only work for isometric base tile, NOT working for orthogonal or hexagonal tile.
-
Tiled map file(.tmx) should use CSV as layer format
-
Each tileset(.tsx) can only have one image (.png) as it source file
-
Not directly support for Animate tile or 2D/3D collider, but it can be made by use extension, see more info on Advance Usage chapter
ITU provide a simple but useful GUI tool. you can try it first
if it's not fit for your scenario.
i suggest you take a look at Advance Usage chapter and write your own one.
ITU will draw all mesh with mesh render on one or more sorting layer .
the GUI part only draw all Tiled layer on one Unity sorting layer .
but if you need draw each layer separately . you can modify it in render setting via script (see more on Advance Usage).
it need to define each sorting layer first before any rendering start.
Edit -> Project Settings -> Tags and Layers
ITU will use ITU_Tiled
by default, you can add ITU_Tiled
or any name you liked in it.
ITU GUI tool can be opened via Tools -> IsometricTiled2Unity -> Import Txm
click the open button , choice a Tiled map file to open
you can put the Raw file (.tmx, .tsx .png) in any folder you liked , but i suggest to put those file outside Unity project folder , because ITU will duplicate those file and copy it under ITU_Porject
folder .
click Choice
button to choice a folder as the ITU_Project
folder.
when ITU process finish , it will generate some sub folder.
$ITU_Project_Folder/Maps/$Tiled_Name/Meshes
: store all mesh for map $Tiled_Name
$ITU_Project_Folder/Maps/$Tiled_Name/Prefab
: the map prefab
$ITU_Project_Folder/Tilesets/Imgs
: tileset source image
$ITU_Project_Folder/Tilesets/Materials
: material use for the same name tileset
after chose Tiled map file and ITU_Project folder ,there will show up some render setting let you play with.
same as Unity pixel pre unit , default is 100
chunk render size define how many tile will combine to one mesh. eg chunk size=10 , it means 10(width) x 10(height) = 100 tile will combine together as one mesh.
Unity automatically culling mesh outside camera. so control the chunk size is the key to gain good performance on huge map.
but be aware , it't not mean use small chunk size is good, it will generate more GameObject when chunk size is too small , and it will also cause performance issue.
if IsStatic = true , it will mark the entire TileMap static, except navigation. because if mark the gameObject navigation static ,it will effected navMesh generate.
ITU will create a tilemap prefab at Path $ITU_Project_Folder/Maps/$Tiled_Name/Prefab
if CreateInstance = true then , when ITU render complete , it will create a instance of this prefab on current scene.
define the material for each tileset, if leave this property to null
, then IUT will use Unlit/Transparent
as default.
it is possible to assign different material for each tileset via script, see more info on Advance Usage chapter
define the sorting layer , it is possible to assign different sorting layer for each Tiled layer via script, see more info on Advance Usage
part
define which Tiled layer you don't want ITU to render
eg, there have three layer (Sky,Building,Ground) defined in Tiled map , and you set Building to ignore layers , then ITU will only render Sky and Ground.
just click the button , and ITU will do the rest for you.
you can create a CustomeEditor, EditorWindow , MenuItem or whatever way you want to run ITU via script.
here is a demo on how to run ITU with code.
public string txmFilePath = "/Plugins/Eran/IsometricTiled2Unity/Example/Res/Raw/Tiled/Maps/ITU_Nav_Prefab_Demo.tmx";
public string saveProjectToPath = "/Plugins/Eran/IsometricTiled2Unity/Example/Res/InGame";
var itu = new ITU_Facade();
//Load Txm map
itu.LoadTxmFile(Application.dataPath + txmFilePath);
//Attach extension
var ext1 = new YourCustomExtension1();
var ext2 = new YourCustomExtension1();
itu.AddExtensions(ext1);
itu.AddExtensions(ext2);
//Render
var rednerSetting = new ITU_RenderSetting();
rednerSetting.SetSaveToFolder(saveProjectToPath, Path.GetFileNameWithoutExtension(txmFilePath));
itu.SetRenderSetting(rednerSetting);
itu.RenderMapToCurrentScene();
//Handle Object & Property
itu.HandleMapPropertyAndObject();
there have four step for the all process.
this is the full path of the Tild map file you want to process. if the file under Unity project folder , you can use Application.dataPath+$relateFilePath
to get full Path.
if the file is outside the project folder , you can define it by const string , or use EditorUtility.OpenFilePanel to select it manually.
var rednerSetting = new ITU_RenderSetting();
rednerSetting.SetSaveToFolder(path,txmFileName);
in renderSetting ,you can use function SetSaveToFolder
to define all saveToPath,
the result would be same as the GUI Windows
//ITU_RenderSetting.cs
public void SetSaveToFolder(string _path, string _txmFileName)
{
meshSaveToPath = _path + "/Maps/" + _txmFileName + "/Meshes";
prefabSaveToPath = _path + "/Maps/" + _txmFileName + "/Prefab";
imgSaveToPath = _path + "/Tilesets/Imgs";
materailSaveToPath = _path + "/Tilesets/Materials";
}
if you prefer to handle it in you own way , you can also set those saveToPath separately
var rednerSetting = new ITU_RenderSetting();
rednerSetting.meshSaveToPath="xxx";
rednerSetting.prefabSaveToPath="xxx";
rednerSetting.imgSaveToPath="xxx";
rednerSetting.materailSaveToPath="xxx";
be aware that
-
all saveToPath is relate to Unity project folder
-
ITU will Delete all mesh file under
rednerSetting.meshSaveToPath
when it start.
most of the render setting property are expose in GUI part, but some of them can only access via script. you can check ITU_RenderSetting.cs
file to see more details.
tilesetMaterials
is use to define custom material for each tileset.
for example you have a animate tile for render water layer (AnimateWaterTileset.tsx)
and you already wrote a shader(AnimateTileShader) to handle that.
then you can add this to tilesetMaterials
.
rednerSetting.tilesetMaterials.Add("AnimateWaterTileset",new Material(Shader.Find("AnimateTileShader")))
IUT default use Unlit/Transparent
for tileset render , if you want change that , you can replace it with other by set defaultMaterial
rednerSetting.defaultMaterial = new Material(Shader.Find("YourShader"));
sometimes you don't want all Tiled layer rendered in one single sorting layer .
e.g you defined three layer in Tiled(Top,Middle,Bottom) and you want use three Unity sorting layer to handle it.
first you need define those sorting layer in Edit->ProjectSetting->Tag&Layer
(IUT_Top,IUT_Middle,IUT_Bottom)
then you can set those via code:
rednerSetting.customSortLayerDic.Add("Top","IUT_Top");
rednerSetting.customSortLayerDic.Add("Middle","IUT_Middle");
rednerSetting.customSortLayerDic.Add("Bottom","IUT_Bottom");
also if ITU can't find any Tiled layer defined in customSortLayerDic
then it will draw that layer to defaultSortLayerName
,
the default named is ITU_Tiled
, you can change it by
rednerSetting.defaultSortLayerName="xxx";
when prepare is done , you can call
var mapMeta = itu.LoadTxmFile($txmFilePath);
to load the Tiled map file.
ITU is not only deserialize the Tiled map file(.txm) and Tild tileset file(.tsx) but also did some link and calculate stuff
eg, you can:
- access each tile uv info by
ITU_TsxEachTileMeta.cs
- access each tileset image full path by
ITU_TsxRootMeta.cs
...
normally you don't need pay attention on this part , but in some peculiar case , you may ignore render and handle object & property part, and just use the raw meta file directly.
in that case i would suggest you set a break point after itu.LoadTxmFile()
and see what ITU already done , and take it from there.
after the render setting is set , you can call
itu.RenderMapToCurrentScene();
to start the render process .
you may use extension to engage the render process , see more info on Working with extension
part.
and the render process is not necessary part, sometimes you may just want use Tiled as a level editor ,export level data to a json file, no render required.
in that case,you can just ignore those part , and jump to Object & Property
handle part.
call
itu.HandleMapPropertyAndObject();
will start the process ,but object & property handle only work with extension, so if you not set any extension to ITU before running this process , nothing will happen.
extension is the best way to engage with ITU . create a C# Class, and extends ITU_ExtensionBase
class , then use
var ext1 = new YourCustomExtension1();
itu.AddExtensions(ext1);
add to ITU before it start any process.
extension have several virtual function you could override
Render part:
public virtual void OnRenderMapProcessBegin(){}
public virtual void OnRenderMapProcessFinish(GameObject _mapGameObject){}
public virtual void OnRenderTile(Vector2Int _tileIndex, Vector2 _tileTopCenterScenePos, float _tileWidthInUnit, float _tileHeightInUnit, ITU_TmxLayerMeta _layerMeta, ITU_TsxEachTileMeta _tileMeta){}
OnRenderMapProcessBegin
will call before render process begin.
OnRenderMapProcessFinish
will call when the render process finish
_mapGameObject
: the render process result.
OnRenderTile
will call when ITU process tile.
_tileIndex
: tile index in Tiled_tileTopCenterScenePos
: tile position in current scene (x-y plane)
_tileWidthInUnit
&_tileHeightInUnit
: unit size of each tile_layerMeta
: current layer meta file_tileMeta
current render tile meta file
Property & Object part:
public virtual void HandleMapPropertyAndObjectBegin(){}
public virtual void HandleMapPropertyAndObjectFinish(){}
public virtual void HandleMapProperty(ITU_PropertieRootMeta _mapProperty){}
public virtual void HandleLayerProperty(ITU_TmxLayerMeta _layerMeta, ITU_PropertieRootMeta _layerProperty){}
public virtual void HandleMapObject(ITU_EachObjectMeta _objectMeta){}
public virtual void HandelTileWithProperty(Vector2Int _tileIndex, Vector2 _tileTopCenterScenePos,float _tileWidthInUnit, float _tileHeightInUnit,
ITU_TmxLayerMeta _layerMeta,
ITU_TsxEachTileMeta _tileMeta){}
same as render part , but those function will only be called when you set property to it.
e.g if you not set any map property , then HandleMapProperty
will not be called.
to help you quick understand how does extension work , i wrote two simple example
in this example i will try to calculate the map basic info and use that info init a monobehavior class ,then attach it to the map gameobject.
you can watch this video(https://youtu.be/qMHvjy4crjU) or open the scene in
/Plugins/Eran/IsometriceTiled2Unity/Example/Demos/ITU_MapInfo
to see more info.
public class ITU_ReadTxmMapInfoExtension : ITU_ExtensionBase
{
private float minPosX = float.MaxValue;
private float minPosY = float.MaxValue;
private float maxPosX = float.MinValue;
private float maxPosY = float.MinValue;
public override void OnRenderTile(Vector2Int _tileIndex, Vector2 _tileTopCenterScenePos, float _tileWidthInUnit, float _tileHeightInUnit, ITU_TmxLayerMeta _layerMeta,
ITU_TsxEachTileMeta _tileMeta)
{
var minX = _tileTopCenterScenePos.x - _tileWidthInUnit / 2f;
var maxX = _tileTopCenterScenePos.x + _tileWidthInUnit / 2f;
var maxY = _tileTopCenterScenePos.y;
var minY = _tileTopCenterScenePos.y - _tileHeightInUnit;
minPosX = Mathf.Min(minPosX, minX);
minPosY = Mathf.Min(minPosY, minY);
maxPosX = Mathf.Max(maxPosX, maxX);
maxPosY = Mathf.Max(maxPosY, maxY);
}
public override void OnRenderMapProcessFinish(GameObject _mapGameObject)
{
var sp = _mapGameObject.AddComponent<TxmMapInfo>();
sp.mapFileName = mMapMeta.fileName;
sp.mapWidthInUnit = Mathf.Abs(maxPosX - minPosX);
sp.mapHeightInUnit = Mathf.Abs(maxPosY - minPosY);
sp.isStaggered = mMapMeta.IsIsoStaggered();
sp.mapLeftTopPosV2 = new Vector2(minPosX, maxPosY);
sp.mapRightBottomPosV2 = new Vector2(maxPosX, minPosY);
}
}
on each time OnRenderTile
called, the extension will compare the min/max value .
and when all render process finish , ITU will call OnRenderMapProcessFinish
, in this function you can attach all calculated info back to the map gameObject.
in this example i will try to replace one layer tile with Prefab ,and generate navMesh for the map.
to achieve this goal, i need use two extension , one work for tile to prefab logic , the other work for generate navmesh
you can watch this video(https://youtu.be/sUwdwxEQlYs) or open the scene in
/Plugins/Eran/IsometriceTiled2Unity/Example/Demos/ITU_Nav_Prefab
to see more info.
Fix the sorting layer issue
the demo scene Demo_Nav_Prefab
put all prefab in a soring layer named GameItem
. but when you import this unitypackage
to your own project , there won't be a sorting layer named as GameItem
, so first we need to fix that issue.
create a new sorting layer in Tag&Layers, make sure this sorting layer is under the ITU output layer, which default is ITU_Tiled
.
then assign the new layer to both SupportDeco
& BoxDeco
, click Apply button.
Tile to Prefab
in Tiled i defined three Layer : Deco Floor Ground
first i need tell ITU that do not render Deco
layer
rednerSetting.ignoreRenderLayerNameList.Add("Deco");
then i need defined some custom property in tileset , so that extension can know which tile need to replace with prefab , and replace to which prefab.
with everything setup completed, then can write the extension :
public override void HandelTileWithProperty(Vector2Int _tileIndex, Vector2 _tileTopCenterScenePos, float _tileWidthInUnit, float _tileHeightInUnit, ITU_TmxLayerMeta _layerMeta,
ITU_TsxEachTileMeta _tileMeta)
{
//Only replace tile in deco layer with prefab
//the support deco is also used in Ground layer,and it will not replace.
if (_layerMeta.layerName != "Deco") return;
if (_tileMeta.property.HasProperty("IsDeco"))
{
var ID = _tileMeta.property.GetProperty("ID").value;
var PrefabDeco = ID == "Box" ? PrefabBoxDeco : PrefabSupportDeco;
var go = PrefabUtility.InstantiatePrefab(PrefabDeco) as GameObject;
go.transform.parent = prefabDecoRoot.transform;
go.transform.localEulerAngles = new Vector3(90, 0, 0);
go.transform.position = new Vector3(_tileTopCenterScenePos.x, 1f, _tileTopCenterScenePos.y - +_tileHeightInUnit / 2f);
}
}
_tileMeta.property.HasProperty("IsDeco")
can check is current tile have IsDeco
key , but it's not check the value
part.
in this example no need to check the value part , because other tile do not contain IsDeco
key , but if you want with the value you can use
_tileMeta.property.GetProperty("IsDeco").value=="true"
to check the bool value.
in ITU, all the property value are store in string , but you can use
property.GetProperty("xxx").type == ITU_EachPropertieMeta.TYPE_INT
to determine the property value type. see more info in class ITU_EachPropertieMeta.cs
Generate NavMesh
same as Tiled 2 Prefab part, first set the IsWalkable property
then the extension can use this property to generate a tile block, also set this block to Navigation Static
.
the rest of part is just standard Unity NavMesh process flow , adjust the baked agent size and bake the NavMesh.