// Copyright (c) Wojciech Figat. All rights reserved. using System; using System.IO; using System.Threading.Tasks; using FlaxEditor.Windows; using FlaxEditor.Windows.Assets; using FlaxEngine; namespace FlaxEditor.Content { /// /// Implementation of for assets. /// /// class CollisionDataItem : BinaryAssetItem { /// public CollisionDataItem(string path, ref Guid id, string typeName, Type type) : base(path, ref id, typeName, type, ContentItemSearchFilter.Other) { } /// public override bool OnEditorDrag(object context) { return true; } /// public override Actor OnEditorDrop(object context) { return new MeshCollider { CollisionData = FlaxEngine.Content.LoadAsync(ID) }; } } /// /// A asset proxy object. /// /// [ContentContextMenu("New/Physics/Collision Data")] class CollisionDataProxy : BinaryAssetProxy { /// public override string Name => "Collision Data"; /// public override EditorWindow Open(Editor editor, ContentItem item) { return new CollisionDataWindow(editor, item as AssetItem); } /// public override AssetItem ConstructItem(string path, string typeName, ref Guid id) { return new CollisionDataItem(path, ref id, typeName, AssetType); } /// public override Color AccentColor => Color.FromRGB(0x2c3e50); /// public override Type AssetType => typeof(CollisionData); /// public override bool CanCreate(ContentFolder targetLocation) { return targetLocation.CanHaveAssets; } /// public override void Create(string outputPath, object arg) { if (Editor.CreateAsset("CollisionData", outputPath)) throw new Exception("Failed to create new asset."); } private bool TryUseCollisionData(Model model, BinaryAssetItem assetItem, Action created, bool alwaysDeferCallback, CollisionDataType type) { var collisionData = FlaxEngine.Content.Load(assetItem.ID); if (collisionData) { var options = collisionData.Options; if ((options.Model == model.ID || options.Model == Guid.Empty) && options.Type == type) { Editor.Instance.Windows.ContentWin.Select(assetItem); if (created != null && alwaysDeferCallback) FlaxEngine.Scripting.InvokeOnUpdate(() => created(collisionData)); else if (created != null) created(collisionData); return true; } } return false; } /// /// Create collision data from model. /// /// The associated model. /// The action to call once the collision data gets created (or reused from existing). /// True if start initial item renaming by user, or tru to skip it. /// True if always call callback on the next engine update, otherwise callback might be called within this function if collision data already exists. /// Type of the collider to create. public void CreateCollisionDataFromModel(Model model, Action created = null, bool withRenaming = true, bool alwaysDeferCallback = true, CollisionDataType type = CollisionDataType.ConvexMesh) { // Check if there already is collision data for that model to reuse var modelItem = (AssetItem)Editor.Instance.ContentDatabase.Find(model.ID); if (modelItem?.ParentFolder != null) { foreach (var child in modelItem.ParentFolder.Children) { // Check if there is collision that was made with this model if (child is BinaryAssetItem b && b.IsOfType()) { if (TryUseCollisionData(model, b, created, alwaysDeferCallback, type)) return; } // Check if there is an auto-imported collision if (child is ContentFolder childFolder && childFolder.ShortName == modelItem.ShortName) { foreach (var childFolderChild in childFolder.Children) { if (childFolderChild is BinaryAssetItem c && c.IsOfType()) { if (TryUseCollisionData(model, c, created, alwaysDeferCallback, type)) return; } } } } } // Create new item so user can name it and then generate collision for it in async Action create = contentItem => { var assetItem = (AssetItem)contentItem; var collisionData = FlaxEngine.Content.LoadAsync(assetItem.ID); if (collisionData == null || collisionData.WaitForLoaded()) { Editor.LogError("Failed to load created collision data."); return; } Task.Run(() => { Editor.CookMeshCollision(assetItem.Path, type, model); if (created != null) FlaxEngine.Scripting.InvokeOnUpdate(() => created(collisionData)); }); }; var initialName = (modelItem?.ShortName ?? Path.GetFileNameWithoutExtension(model.Path)) + " Collision"; Editor.Instance.Windows.ContentWin.NewItem(this, null, create, initialName, withRenaming); } } }