using UnityEngine;
using UnityEngine.U2D;
using System.Collections.Generic;
using UnityEditor.SceneManagement;
using UnityEditor.U2D.SpriteShape;
using Object = UnityEngine.Object;

namespace UnityEditor.U2D
{
    [InitializeOnLoad]
    internal static class SceneDragAndDrop
    {
        static SceneDragAndDrop()
        {
#if UNITY_2019_1_OR_NEWER
            SceneView.duringSceneGui += OnSceneGUI;
            EditorApplication.hierarchyWindowItemOnGUI += OnHierarchyGUI;
#else
            SceneView.onSceneGUIDelegate += OnSceneGUI;
#endif
        }

        static class Contents
        {
            public static readonly string createString = "Create Sprite Shape";
        }

        static List<Object> s_SceneDragObjects;
        static DragType s_DragType;
        enum DragType { NotInitialized, CreateMultiple }

        public delegate string ShowFileDialogDelegate(string title, string defaultName, string extension, string message, string defaultPath);

        static void OnSceneGUI(SceneView sceneView)
        {
            HandleSceneDrag(sceneView, Event.current, DragAndDrop.objectReferences, DragAndDrop.paths);
        }

        public static GameObject Create(UnityEngine.U2D.SpriteShape shape, Vector3 position, SceneView sceneView)
        {
            string name = string.IsNullOrEmpty(shape.name) ? "New SpriteShapeController" : shape.name;
            name = GameObjectUtility.GetUniqueNameForSibling(null, name);
            GameObject go = new GameObject(name);

            SpriteShapeController shapeController = go.AddComponent<SpriteShapeController>();
            shapeController.spriteShape = shape;
            go.transform.position = position;
            go.hideFlags = HideFlags.HideAndDontSave;

            return go;
        }

        static void OnHierarchyGUI(int instanceID, Rect rect)
        {
            HandleSceneDrag(null, Event.current, DragAndDrop.objectReferences, null);
        }

        static List<UnityEngine.U2D.SpriteShape> GetSpriteShapeFromPathsOrObjects(Object[] objects, string[] paths, EventType currentEventType)
        {
            List<UnityEngine.U2D.SpriteShape> result = new List<UnityEngine.U2D.SpriteShape>();

            foreach (Object obj in objects)
            {
                if (AssetDatabase.Contains(obj))
                {
                    if (obj is UnityEngine.U2D.SpriteShape)
                        result.Add(obj as UnityEngine.U2D.SpriteShape);
                }
            }
            return result;
        }

        static void HandleSceneDrag(SceneView sceneView, Event evt, Object[] objectReferences, string[] paths)
        {
            if (evt.type != EventType.DragUpdated && evt.type != EventType.DragPerform && evt.type != EventType.DragExited)
                return;

            switch (evt.type)
            {
                case EventType.DragUpdated:
                    {
                        DragType newDragType = DragType.CreateMultiple;

                        if (s_DragType != newDragType || s_SceneDragObjects == null)
                        // Either this is first time we are here OR evt.alt changed during drag
                        {
                            if (ExistingAssets(objectReferences))     // External drag with images that are not in the project
                            {
                                List<UnityEngine.U2D.SpriteShape> assets = GetSpriteShapeFromPathsOrObjects(objectReferences, paths,
                                        evt.type);

                                if (assets.Count == 0)
                                    return;

                                if (s_DragType != DragType.NotInitialized)
                                    // evt.alt changed during drag, so we need to cleanup and start over
                                    CleanUp(true);

                                s_DragType = newDragType;
                                CreateSceneDragObjects(assets);
                            }
                        }

                        if (s_SceneDragObjects != null)
                        {
                            if (sceneView != null)
                                PositionSceneDragObjects(s_SceneDragObjects, sceneView, evt.mousePosition);

                            DragAndDrop.visualMode = DragAndDropVisualMode.Copy;
                            evt.Use();
                        }
                    }
                    break;
                case EventType.DragPerform:
                    {
                        List<UnityEngine.U2D.SpriteShape> assets = GetSpriteShapeFromPathsOrObjects(objectReferences, paths, evt.type);

                        if (assets.Count > 0 && s_SceneDragObjects != null)
                        {
                            // For external drags, we have delayed all creation to DragPerform because only now we have the imported sprite assets
                            if (s_SceneDragObjects.Count == 0)
                            {
                                CreateSceneDragObjects(assets);
                                if (sceneView != null)
                                    PositionSceneDragObjects(s_SceneDragObjects, sceneView, evt.mousePosition);
                            }

                            var prefabStage = PrefabStageUtility.GetCurrentPrefabStage();

                            foreach (GameObject dragGO in s_SceneDragObjects)
                            {
                                if (prefabStage != null)
                                {
                                    var parTransform = Selection.activeTransform != null
                                        ? Selection.activeTransform
                                        : prefabStage.prefabContentsRoot.transform;
                                    dragGO.transform.SetParent(parTransform, true);
                                }

                                Undo.RegisterCreatedObjectUndo(dragGO, "Create Shape");
                                dragGO.hideFlags = HideFlags.None;
                            }

                            Selection.objects = s_SceneDragObjects.ToArray();

                            CleanUp(false);
                            evt.Use();
                        }
                    }
                    break;
                case EventType.DragExited:
                    {
                        if (s_SceneDragObjects != null)
                        {
                            CleanUp(true);
                            evt.Use();
                        }
                    }
                    break;
            }
        }

        static void PositionSceneDragObjects(List<Object> objects, SceneView sceneView, Vector2 mousePosition)
        {
            Vector3 position = Vector3.zero;
            position = HandleUtility.GUIPointToWorldRay(mousePosition).GetPoint(10);
            if (sceneView.in2DMode)
            {
                position.z = 0f;
            }
            else
            {
                DragAndDrop.visualMode = DragAndDropVisualMode.Copy;
                object hit = HandleUtility.RaySnap(HandleUtility.GUIPointToWorldRay(mousePosition));
                if (hit != null)
                {
                    RaycastHit rh = (RaycastHit)hit;
                    position = rh.point;
                }
            }

            foreach (GameObject gameObject in objects)
            {
                gameObject.transform.position = position;
            }
        }

        static void CreateSceneDragObjects(List<UnityEngine.U2D.SpriteShape> shapes)
        {
            if (s_SceneDragObjects == null)
                s_SceneDragObjects = new List<Object>();

            if (s_DragType == DragType.CreateMultiple)
            {
                foreach (UnityEngine.U2D.SpriteShape sprite in shapes)
                    s_SceneDragObjects.Add(CreateDragGO(sprite, Vector3.zero));
            }
            else
            {
                s_SceneDragObjects.Add(CreateDragGO(shapes[0], Vector3.zero));
            }
        }

        static void CleanUp(bool deleteTempSceneObject)
        {
            if (deleteTempSceneObject)
            {
                foreach (GameObject gameObject in s_SceneDragObjects)
                    Object.DestroyImmediate(gameObject, false);
            }

            if (s_SceneDragObjects != null)
            {
                s_SceneDragObjects.Clear();
                s_SceneDragObjects = null;
            }

            s_DragType = DragType.NotInitialized;
        }

        static bool ExistingAssets(Object[] objects)
        {
            foreach (Object obj in objects)
            {
                if (AssetDatabase.Contains(obj))
                    return true;
            }
            return false;
        }

        static GameObject CreateDragGO(UnityEngine.U2D.SpriteShape spriteShape, Vector3 position)
        {
            SpriteShapeController spriteShapeController = SpriteShapeEditorUtility.CreateSpriteShapeController(spriteShape);
            GameObject gameObject = spriteShapeController.gameObject;
            gameObject.transform.position = position;
            gameObject.hideFlags = HideFlags.HideAndDontSave;
            spriteShapeController.spriteShape = spriteShape;

            SpriteShapeEditorUtility.SetShapeFromAsset(spriteShapeController);

            return gameObject;
        }
    }
}