Do you ever think to use enum for making ID because it can be access the from code?
For long list of item id this is not really a good solution, but for short id this should be fine.
ScriptableObject
ScriptableObject is used for the user interface of this generator because the user most likely wanna have more than one enum generator they can edit. In this ScriptableObject, we need to define directory, namespace, enum name and values.
Generator
There are two ways to generate this enum,
- Create a template file.
- Using C# CodeDom.
Using CodeDom the generated code will be automatically indented and there should be no error in the indentation or spacing.
This code generator is also can be used for unit testing, but it’s on another level.
For this simple generator, we just need to create,
- CodeNamespace, represent a namespace declaration.
- CodeTypeDeclaration, this is where we can declare enum, class, interface and struct.
- CodeCompileUnit, a container for the CodeDom program graph or the above variable.
private readonly CodeNamespace _codeNamespace = new();
private readonly CodeTypeDeclaration _codeTypeDeclaration = new();
private CodeCompileUnit _codeCompileUnit = new();
public EnumGenerationSystem()
{
_codeTypeDeclaration.IsEnum = true;
_codeNamespace.Types.Add(_codeTypeDeclaration);
_codeCompileUnit.Namespaces.Add(_codeNamespace);
}
Use CodeMemberField to add the enum values. CodeMemberField is used to declare a field or type.
foreach (var enumValue in Values)
{
CodeMemberField field = new CodeMemberField(EnumName, enumValue);
_codeTypeDeclaration.Members.Add(field);
}
And finally, write the code into a C# file.
CodeDomProvider provider = CodeDomProvider.CreateProvider("CSharp");
CodeGeneratorOptions options = new CodeGeneratorOptions
{
BracingStyle = "C"
};
using (StreamWriter sourceWriter = new StreamWriter(filePath))
{
provider.GenerateCodeFromCompileUnit(_codeCompileUnit, sourceWriter, options);
AssetDatabase.Refresh();
}
Trigger
For the trigger to generate the enum, we use ContextMenu
, not pretty convenient compared to a button, but at least we can use it. Later we can update it to use a button or even a custom editor window, but I stick to ScriptableObject because this is the faster way to get the generator working.
[ContextMenu("Generate")]
public void GenerateEnum()
{
EnumGenerationSystem enumGenerationSystem = new EnumGenerationSystem
{
Namespace = nameSpace,
EnumName = enumName,
FileDirectory = Application.dataPath + "/" + filePath,
Values = values
};
enumGenerationSystem.GenerateEnum();
}
Result
Below is the generated enum from the ScriptableObject above.
//------------------------------------------------------------------------------
// <auto-generated>
// This code was generated by a tool.
// Runtime Version:4.0.30319.42000
//
// Changes to this file may cause incorrect behavior and will be lost if
// the code is regenerated.
// </auto-generated>
//------------------------------------------------------------------------------
namespace NoodleEater.Toolset.EnumGenerator
{
public enum ItemTier
{
Tier1,
Tier2,
Tier3,
Tier4,
Tier5,
}
}
You can see the source code here
Thank you 😊 I hope you enjoy it.