From aa31639ec92decdc8032ed58390bd790f0dd686d Mon Sep 17 00:00:00 2001 From: pilorenzo <60971887+pilorenzo@users.noreply.github.com> Date: Mon, 18 Sep 2023 11:33:49 +0200 Subject: [PATCH 1/7] Update Exporter.cs Solved issue with position scale and rotation of nodes --- Exporter/Exporter.cs | 17 +++++++++++------ 1 file changed, 11 insertions(+), 6 deletions(-) diff --git a/Exporter/Exporter.cs b/Exporter/Exporter.cs index 21a6949..96c856e 100644 --- a/Exporter/Exporter.cs +++ b/Exporter/Exporter.cs @@ -1,4 +1,4 @@ -using System.Collections.Generic; +using System.Collections.Generic; using System.IO; using System.Reflection; using UnityEditor; @@ -12,7 +12,7 @@ class Exporter : IConvertedResourceProvider [MenuItem("Godot/Export to Godot 3.1")] static void ExportToGodot31() { - var e = new Exporter("D:/PROJETS/INFO/UNITY/LD32_EatAndCopulate/GodotProject"); + var e = new Exporter("Godot/GodotProject"); e.Export(); } @@ -134,7 +134,7 @@ Node ExtractNodeBranch(GameObject go) Node rootNode = null; Rigidbody2D promoteRigidBody2D = null; - float rescale2d = 1f; + float rescale2d = 100f; int scriptCount = 0; Script script = null; Dictionary scriptVariables = null; @@ -246,16 +246,21 @@ Node ExtractNodeBranch(GameObject go) } } - node2d.position = go.transform.position * rescale2d; + node2d.position = go.transform.localPosition * rescale2d; node2d.position.y *= -1f; // TODO Invert Y properly // TODO Select pivot //node2d.position.y = Screen.height - node2d.position.y; node2d.scale = go.transform.localScale; + SpriteRenderer goSprite = go.GetComponent(); + if(goSprite) + { + if(goSprite.flipX) node2d.scale.x *= -1f; + if(goSprite.flipY) node2d.scale.y *= -1f; + } - // TODO Is eulerAngles in degrees or radians?? - node2d.rotation = go.transform.rotation.eulerAngles.z; + node2d.rotation = -go.transform.localRotation.eulerAngles.z * Mathf.Deg2Rad; node2d.visible = go.activeSelf; From f4a2f800f0b21ef3d5ce628d0b0b21bbb979908a Mon Sep 17 00:00:00 2001 From: pilorenzo <60971887+pilorenzo@users.noreply.github.com> Date: Mon, 18 Sep 2023 11:35:52 +0200 Subject: [PATCH 2/7] Update Util.cs Directory created before trying to write on it --- Exporter/Util.cs | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/Exporter/Util.cs b/Exporter/Util.cs index 7d13b2c..a393066 100644 --- a/Exporter/Util.cs +++ b/Exporter/Util.cs @@ -1,4 +1,4 @@ -using System; +using System; using System.IO; using System.Text; @@ -8,6 +8,10 @@ static class Util { public static StreamWriter CreateStreamWriter(string path) { + + // Create missing directories + Directory.CreateDirectory(Path.GetDirectoryName(path)); + // Godot expects files to be UTF-8, without BOM, and Unix line endings StreamWriter sw = new StreamWriter(path, false, new UTF8Encoding(false)); sw.NewLine = "\n"; From 0f52e5fb9d5ce22460d0dd3d2424975525bc9a53 Mon Sep 17 00:00:00 2001 From: pilorenzo <60971887+pilorenzo@users.noreply.github.com> Date: Fri, 22 Sep 2023 18:55:05 +0200 Subject: [PATCH 3/7] Update Exporter.cs Corrected atlas region in texture, flipH and flipV in Sprite now match flipX and flipY of Unity SpriteRenderer --- Exporter/Exporter.cs | 18 ++++++++++++------ 1 file changed, 12 insertions(+), 6 deletions(-) diff --git a/Exporter/Exporter.cs b/Exporter/Exporter.cs index 96c856e..b444cc1 100644 --- a/Exporter/Exporter.cs +++ b/Exporter/Exporter.cs @@ -139,6 +139,7 @@ Node ExtractNodeBranch(GameObject go) Script script = null; Dictionary scriptVariables = null; + foreach(var cmp in components) { Node node = null; @@ -155,6 +156,8 @@ Node ExtractNodeBranch(GameObject go) { dst.texture = Util.Cast(ConvertUnityAsset(src.sprite)); rescale2d = src.sprite.pixelsPerUnit; + dst.flipH = src.flipX; + dst.flipV = src.flipY; } } else if(cmp is Rigidbody2D) @@ -248,22 +251,22 @@ Node ExtractNodeBranch(GameObject go) node2d.position = go.transform.localPosition * rescale2d; node2d.position.y *= -1f; - // TODO Invert Y properly - // TODO Select pivot - //node2d.position.y = Screen.height - node2d.position.y; + node2d.scale = go.transform.localScale; SpriteRenderer goSprite = go.GetComponent(); if(goSprite) { - if(goSprite.flipX) node2d.scale.x *= -1f; - if(goSprite.flipY) node2d.scale.y *= -1f; + // if(goSprite.flipX) node2d.scale.x *= -1f; + // if(goSprite.flipY) node2d.scale.y *= -1f; + node2d.zIndex = goSprite.sortingOrder; } node2d.rotation = -go.transform.localRotation.eulerAngles.z * Mathf.Deg2Rad; node2d.visible = go.activeSelf; + rootNode = node2d; } else if(rootCategory == NodeCategory.Control) @@ -317,6 +320,8 @@ Node ExtractNodeBranch(GameObject go) return rootNode; } + + NodeCategory DetectCategory(Component[] components) { int filteredCount = 0; @@ -440,7 +445,8 @@ Resource ConvertSpriteAsset(UnityEngine.Sprite sprite) var at = new AtlasTexture(); at.resourcePath = relPath; at.atlas = atlas; - at.region = sprite.textureRect; + // at.region = sprite.textureRect; + at.region = new Rect(0, 0, sprite.texture.width, sprite.texture.height); // Invert Y at.region.y = sprite.texture.height - at.region.y - at.region.height; From 4129587ef6e92df64b1c0c641b7dd4f566349ffa Mon Sep 17 00:00:00 2001 From: pilorenzo <60971887+pilorenzo@users.noreply.github.com> Date: Mon, 25 Sep 2023 11:06:20 +0200 Subject: [PATCH 4/7] Update Nodes.cs --- Exporter/Nodes.cs | 26 +++++++++++++++++++++++++- 1 file changed, 25 insertions(+), 1 deletion(-) diff --git a/Exporter/Nodes.cs b/Exporter/Nodes.cs index a9cea2f..b5daf38 100644 --- a/Exporter/Nodes.cs +++ b/Exporter/Nodes.cs @@ -1,4 +1,4 @@ -using System.Collections.Generic; +using System.Collections.Generic; using UnityEngine; namespace Godot @@ -41,11 +41,17 @@ class CanvasItem : Node public Color selfModulate = Color.white; public bool visible = true; + public bool zRelative = true; + public int zIndex = 0; + public override Dictionary GetData() { var d = base.GetData(); d.Add("self_modulate", selfModulate); d.Add("visible", visible); + d.Add("z_as_relative", zRelative); + d.Add("z_index", zIndex); + return d; } } @@ -69,15 +75,33 @@ public override Dictionary GetData() class Sprite : Node2D { public Texture texture; + public bool flipH = false; + public bool flipV = false; public override Dictionary GetData() { var d = base.GetData(); d.Add("texture", texture); + d.Add("flip_h", flipH); + d.Add("flip_v", flipV); return d; } } + + class ParallaxLayer : Node2D + { + public Vector2 motionScale = Vector2.one; + + public override Dictionary GetData() + { + var d = base.GetData(); + d.Add("motion_scale", motionScale); + return d; + } + } + + class CollisionObject2D : Node2D { } From e9f30bc9bba8038ebd643e6ecdcb7cf9d1d03543 Mon Sep 17 00:00:00 2001 From: pilorenzo <60971887+pilorenzo@users.noreply.github.com> Date: Sun, 19 Nov 2023 19:16:20 +0100 Subject: [PATCH 5/7] Canvas layer support for fake 3d (see comment in code) --- Exporter/Exporter.cs | 30 ++++++++++++++++++++++++++++-- 1 file changed, 28 insertions(+), 2 deletions(-) diff --git a/Exporter/Exporter.cs b/Exporter/Exporter.cs index b444cc1..6237f21 100644 --- a/Exporter/Exporter.cs +++ b/Exporter/Exporter.cs @@ -1,6 +1,7 @@ using System.Collections.Generic; using System.IO; using System.Reflection; + using UnityEditor; using UnityEngine; using UnityEngine.SceneManagement; @@ -105,7 +106,7 @@ void ConvertScene(Scene scene) var node = ExtractNodeBranch(go); if(node != null) { - node.name = go.name; + // node.name = go.name; tree.AddChild(node); } } @@ -123,7 +124,7 @@ enum NodeCategory Control } - Node ExtractNodeBranch(GameObject go) + Node ExtractNodeBranch(GameObject go, bool evaluateCanvasLayer = true) { // TODO Detect if go is a prefab with PrefabUtility @@ -207,6 +208,31 @@ Node ExtractNodeBranch(GameObject go) } else { + /* + Creates a canvas layer as a parent node for the layer, + to create a fake 3d effect in Godot. + Each layer must be contained by an empty object + named LayerX, where X is the layer number + (ex. Layer0, Layer-1, Layer+2, Layer3). + If X is not a number the layer number is 0. + The z position of the empty object LayerX is used to determine + the parallax effect (follow_viewport_scale()) for all the layer's children. + The scale of LayerX must be (1,1,1) to + obtain an equal parallax effect. + */ + if(go.name.Contains("Layer") && evaluateCanvasLayer) + { + var canvasLayer = new CanvasLayer() + { + name = "Canvas" + go.name, + layer = int.TryParse(go.name.TrimStart("Layer".ToCharArray()), out int nbr)? nbr:0, + fwScale = 1f + -0.025f*go.transform.position.z + }; + rootNode = canvasLayer; + Node childNode = ExtractNodeBranch(go, false); + rootNode.AddChild(childNode); + return rootNode; + } // TODO more } From 9753ed441bf24e84c53d00b955b9170c9c7c1700 Mon Sep 17 00:00:00 2001 From: pilorenzo <60971887+pilorenzo@users.noreply.github.com> Date: Sun, 19 Nov 2023 19:17:48 +0100 Subject: [PATCH 6/7] Added CanvasLayer node --- Exporter/Nodes.cs | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/Exporter/Nodes.cs b/Exporter/Nodes.cs index b5daf38..5034f7a 100644 --- a/Exporter/Nodes.cs +++ b/Exporter/Nodes.cs @@ -36,6 +36,25 @@ public void AddChild(Node newChild) } } + + + class CanvasLayer : Node + { + public int layer = 0; + public bool fwEnabled = true; + public float fwScale = 1.0f; + public override Dictionary GetData() + { + var d = base.GetData(); + d.Add("layer", layer); + d.Add("follow_viewport_enabled", fwEnabled); + d.Add("follow_viewport_scale", fwScale); + return d; + } + + } + + class CanvasItem : Node { public Color selfModulate = Color.white; @@ -56,6 +75,8 @@ public override Dictionary GetData() } } + + class Node2D : CanvasItem { public Vector2 position; From 41fc1899843b420dc7306b7ef11bdf14644726d6 Mon Sep 17 00:00:00 2001 From: pilorenzo <60971887+pilorenzo@users.noreply.github.com> Date: Sun, 19 Nov 2023 19:53:31 +0100 Subject: [PATCH 7/7] Update README.md --- README.md | 47 +++++++---------------------------------------- 1 file changed, 7 insertions(+), 40 deletions(-) diff --git a/README.md b/README.md index d2d0fbd..94232e4 100644 --- a/README.md +++ b/README.md @@ -1,11 +1,14 @@ Unity Engine to Godot Engine exporter ======================================= -This is an experimental script that allows you to convert all scenes in your Unity project into a Godot project. It is not aimed at doing everything automatically, only things that can be converted decently. -It's only a proof of concept on simple 2D games for now, and a ton of work remains to be done if it were to support everything else. -While there are always cases where conversion is ambiguous and things to do manually, it's still fun to at least have the ability to automate this to some extent. +This is a tool designed to convert 2D projects from Unity to Godot, based on [Zylann's unity_to_godot_converter](https://github.com/Zylann/unity_to_godot_converter). -I have other projects to work on so I won't work much on this tool for now, and I am aware that there is an abysmal amount of features it could support^^ But feel free to hack around with it and improve it if you like the idea. +I have just made some modifications to: + +1. Properly restore the position of sprites in the scene. +2. Recreate the same parallax effect achieved in Unity by considering the sprite's z-axis position, utilizing CanvasLayer (see [Exporter.cs](https://github.com/pilorenzo/unity_to_godot_converter/blob/master/Exporter/Exporter.cs)). + +Tested on Godot 4.1.3 (only sprites and empty objects) How to install @@ -14,39 +17,3 @@ How to install Copy this repository in your Unity project, inside a folder named `Editor`, and you should see a new `Godot` menu with options in it. Although it should not modify anything in the project, it's up to you to preserve your data if anything wrong happens :p - - -Some challenges ------------------ - -Here is a random list of things I had to take choices, for which workarounds may or may not exist. -There may be a lot more, but you can get an idea of what this tool has to get through: - -- Unity only has `Camera`, but Godot has `Camera2D` and `Camera` for 3D. Choosing which one is ambiguous, so for now I create the 2D version of the camera is orthographic AND if a hint is enabled in the exporter for 2D projects. Also, in Unity, cameras also act as viewports, which is another separate node in Godot, so I'm not sure how to even convert those. Other components are ambiguous too, such as `Light`. - -- Godot has separate engines for 2D and 3D, but Unity only has 3D transforms with ortho camera. So the tool tries to guess what usage a GameObject is for by looking at its components. For example, if it has `SpriteRenderer`, or any of its children does, then the GameObject is converted to a `Node2D`. Otherwise, it becomes a `Spatial`. In some cases, it becomes a blank `Node` in cases where dimensions are irrelevant. - -- Unity uses components attached to GameObjects for its functionality, but Godot uses a node tree directly. That means a single GameObject with several components may convert into one node and several child nodes. If a GameObject only has a Transform and one component, a shortcut is taken to only produce a single Godot node, eliminating the unnecessary nesting. - -- Unity defines rigidbodies as components, but in Godot it is recommended to have such bodies as parent nodes because they control the position of their children, so instead of adding `RigidBodies` as a child nodes, they are have to be promoted as parent. - -- Unity can have multiple scripts on the same GameObject, but Godot can only have one per node. So the converter takes the first script it finds to the root node, and create children `Nodes` for each additional script. You may have to have a manual look after conversion if you use composition a lot. - -- Converting scripts is very complicated, so the tool rather creates stub scripts for each of them so it can still attach them to the proper final nodes, and attempts to preserve serialized variables. For example, when converting to GDScript, a C# script will be parsed for its variables which will be written as `export` on top of an empty GDScript, and the rest or the original source code is written as a big comment below them. This allows to keep configurations and keep track of what the script should be. - -- In Sprite texture resources, Unity allows to define a scaling between pixel coordinates and world coordinates, which is 100 by default, making sprites very small. Godot uses pixels as units at all times, so the plugin attempts to undo this scaling. - -- Unity can subdivide a 2D texture into sprites, so this almost always translates to Godot as `AtlasTextures`. - -- Unity uses a left-handed coordinate system, and in 2D its Y axis stays upwards. In Godot, the Y axis in 2D is downwards, so the tool attempts to invert positions (not working as best as it could at the moment) - -- Godot has no terrain system as of now, but a plugin exists for heightmaps which does not require recompilation. So the plugin could be packaged in the output project, and Unity terrains could be mostly converted to that format. - -- Things requiring a recompilation of Godot cannot be supported, for example the Admob module needs to be integrated into Godot manually by recompiling the engine. - -- Unity and Godot both support prefabs and nested prefabs, but I haven't worked in this part yet. On Unity side it should be a matter of using `PrefabUtility` to detect if a game object is actually an instance of a prefab, and it needs some research to see which delta-modifications are supported both by Unity and Godot. - -- As of 3.1 Godot only saves non-default values in scene data, but this tool can't afford to know them all, so scenes generated by it may be larger than if you had created them in Godot. Saving them from Godot might get rid of the redundancy. - -- Unity can imports 3D models as "fixed" prefabs, a bit like Godot does, so I am not sure if the tool should generate scenes for those, or let Godot do it -