#if UNITY_2019_1_OR_NEWER using System; using System.Collections.Generic; using System.IO; using UnityEditor; using UnityEngine; using UnityAtoms.Extensions; namespace UnityAtoms.Editor { internal class Generator { public void Generate(string type, string baseWritePath, bool isEquatable, List atomTypesToGenerate) { // TODO: More validation of that the type exists / is correct. if (string.IsNullOrEmpty(type)) { Debug.LogWarning($"{Runtime.Constants.LOG_PREFIX} You need to specify a type name. Aborting!"); return; } if (string.IsNullOrEmpty(baseWritePath)) { Debug.LogWarning($"{Runtime.Constants.LOG_PREFIX} You need to specify a write path. Aborting!"); return; } Debug.Log($"{Runtime.Constants.LOG_PREFIX} Generating " + type); // Create directories in path if they don't exists Directory.CreateDirectory(baseWritePath); // Recursively search for template files. TODO: Is there a better way to find and load templates? var templateSearchPath = Runtime.IsUnityAtomsRepo ? Directory.GetParent(Directory.GetParent(Application.dataPath).FullName).FullName : Directory.GetParent(Application.dataPath).FullName; var templatePaths = Directory.GetFiles(templateSearchPath, "UA_Template*.txt", SearchOption.AllDirectories); var templateConditions = isEquatable ? new List() { "EQUATABLE" } : new List(); var capitalizedType = Capitalize(type); var templateVariables = new Dictionary() { { "TYPE_NAME", capitalizedType }, { "TYPE", type } }; foreach (var templatePath in templatePaths) { var templateNameStartIndex = templatePath.LastIndexOf(Path.DirectorySeparatorChar) + 1; var fileExtLength = 4; var templateName = templatePath.Substring(templateNameStartIndex, templatePath.Length - templateNameStartIndex - fileExtLength); var lastIndexOfDoubleUnderscore = templateName.LastIndexOf("__"); var atomType = templateName.Substring(lastIndexOfDoubleUnderscore + 2); var capitalizedAtomType = Capitalize(atomType); var typeOccurrences = templateName.Substring(lastIndexOfDoubleUnderscore - 1, 1).ToInt(def: 1); if (ShouldSkipTemplate(atomTypesToGenerate, capitalizedAtomType, typeOccurrences)) { continue; } var template = File.ReadAllText(templatePath); // Create write directory var dirPath = ResolveDirPath(baseWritePath, capitalizedAtomType, templateName, atomType); Directory.CreateDirectory(dirPath); // Adjust content var content = ResolveVariables(templateVariables, template); content = Templating.ResolveConditionals(template: content, trueConditions: templateConditions); // Write to file var fileName = ResolveFileName(templateVariables, templateName, lastIndexOfDoubleUnderscore, capitalizedType, capitalizedAtomType, typeOccurrences); var filePath = Path.Combine(dirPath, fileName); File.WriteAllText(filePath, content); AssetDatabase.ImportAsset(filePath); } AssetDatabase.Refresh(); } private static string ResolveFileName(Dictionary templateVariables, string templateName, int lastIndexOfDoubleUnderscore, string capitalizedType, string capitalizedAtomType, int typeOccurrences) { if (templateName.Contains("Set{TYPE_NAME}VariableValue")) { return ResolveVariables(templateVariables, $"{capitalizedAtomType}.cs"); } string fileName; if (templateName.Contains("AtomDrawer")) { fileName = $"{capitalizedType.Repeat(typeOccurrences)}{capitalizedAtomType}Drawer.cs"; } else if (templateName.Contains("AtomEditor")) { fileName = $"{capitalizedType.Repeat(typeOccurrences)}{capitalizedAtomType}Editor.cs"; } else { fileName = $"{capitalizedType.Repeat(typeOccurrences)}{capitalizedAtomType}.cs"; } return ResolveVariables(templateVariables, fileName); } private static string ResolveDirPath(string baseWritePath, string capitalizedAtomType, string templateName, string atomType) { if (templateName.Contains("AtomDrawer")) { return Path.Combine(baseWritePath, "Editor", Runtime.IsUnityAtomsRepo ? "Drawers" : "AtomDrawers", $"{atomType}s"); } else if (templateName.Contains("AtomEditor")) { return Path.Combine(baseWritePath, "Editor", Runtime.IsUnityAtomsRepo ? "Editors" : "AtomEditors", $"{atomType}s"); } else if (capitalizedAtomType.Contains("Set{TYPE_NAME}VariableValue")) { return Path.Combine(baseWritePath, "Actions", "SetVariableValue"); } return Path.Combine(baseWritePath, $"{capitalizedAtomType}s"); } private static string ResolveVariables(Dictionary templateVariables, string toResolve) { var resolvedString = toResolve; foreach (var kvp in templateVariables) { resolvedString = resolvedString.Replace("{" + kvp.Key + "}", kvp.Value); } return resolvedString; } private static string Capitalize(string s) { if (string.IsNullOrEmpty(s)) return string.Empty; char[] a = s.ToCharArray(); a[0] = char.ToUpper(a[0]); return new string(a); } private static bool ShouldSkipTemplate(List atomTypesToGenerate, string capitalizedAtomType, int typeOccurrences) { return !atomTypesToGenerate.Exists((a) => a.Type == capitalizedAtomType && a.TypeOccurences == typeOccurrences); } } } #endif