如何将 ScriptableObject 用于关卡的预制转换?
How to use a ScriptableObject for a level's Prefab transforms?
我有 3 种预制件:A、B 和 C
在设计关卡时,在播放模式下,我将它们添加并放置在关卡中。
如何创建一个 ScriptableObject 来保存对这些预制件及其转换的所有实例的引用?
具体来说,在 PlayMode 中,Scriptable 对象应动态响应 PlayMode 中预制件的位置和旋转变化。
我无法想象如何做到这一点,尽管这似乎是对脚本对象的一个很好的使用。
您可以将所需的信息存储在专用 class 中,例如
[Serializable]
public class InstanceInformation
{
public GameObject UsedPrefab;
public Vector3 Position;
public Quaternion Rotation;
public Vector3 Scale;
public InstanceInformation(GameObject usedPrefab, Transform transform)
{
UsedPrefab = usedPrefab;
Rotation = transform.rotation;
Position = transform.position;
Scale = transform.localScale;
}
public void UpdateValues(Transform transform)
{
Rotation = transform.rotation;
Position = transform.position;
Scale = transform.localScale;
}
}
并且在您的 ScriptableObject 中有一个
[CreateAssetMenu]
public class LevelData : ScriptableObject
{
public List<InstanceInformation> instances = new List<InstanceInformation>();
}
然后每次实例化一个预制件时,您也会在其中创建一个相应的条目 instances
。
所以稍后在您的管理器脚本中实例化您所做的事情,例如
// Reference this via the Inspector
public LevelData scriptable;
// For keeping a link for the currently Instantiated stuff
// so everytime you manipulate them later on you can update the according data entry
private Dictionary<GameObject, InstanceInformation> objectToInstanceInformation = new Dictionary<GameObject, InstanceInformation>();
...
var obj = Instantiate(aPrefab, aPosition, aRotation);
var instanceInfo = new InstanceInfo(aPrefab, obj.transform);
// Add the reference to the ScriptableObject list
scriptable.instances.Add(instanceInfo);
// And also keep track of the reference linked to the actual instance
objectToInstanceInformation.Add(obj, instanceInfo);
现在您可以重复或在某个时刻调用
public void SaveInstanceInformations()
{
foreach(var kvp in objectToInstanceInformation)
{
var obj = kvp.key;
var instanceInfo = kvp.value;
instanceInfo.UpdateValues(obj.transform);
}
}
因为 InstanceInformation
是一个 class,因此 reference-type 自动更改此处的值也会更改 ScriptableObject 中 instances
中的相应条目!
所以稍后当你想加载状态时你可以简单地做
foreach (var instance in scriptable.instances)
{
var obj = Instantiate(instance.UsedPrefab, instance.Position, instance.Rotation);
obj.transform.localScale = instance.Scale;
objectToInstanceInformation.Add(obj, instance);
}
最后,为了能够持久存储此数据,您可以例如使用 BinaryFormatter
喜欢
[CreateAssetMenu]
public class LevelData : ScriptableObject
{
// here you can set a filename the data shall be stored to
[SerializeField] private string fileName = "level.dat";
public List<InstanceInformation> instances = new List<InstanceInformation>();
// called when the object is loaded
private void OnEnable()
{
// try to load the data from drive
// in the editor use the Application.streamingAssetsPath
// in a build use Application.persistentDataPath
var folder = Application.isEditor ? Application.streamingAssetsPath : Application.persistentDataPath;
if(!File.Exists(Path.Combine(folder, fileName)))
{
// on the first run the folder and file will not exist
// in the editor do nothing (the file simply doesn't exist yet)
// in a build copy the content from the streaming assets folder to the persistent data path (if exists)
var fallbackFolder = Application.streamingAssetsPath;
if(!Driectoy.Exists(fallbackFolder)) return;
if(!File.Exists(Path.Combine(fallbackFolder, fileName))) return;
// copy fallback file to persistent file
File.Copy(Path.Combine(fallbackFolder, FileName), Path.Combine(folder, fileName));
}
// load the list
using(var file = File.Open(Path.Combine(folder, fileName), FileMode.Open, FileAccess.Read, FileShare.Read))
{
var binaryFormatter = new BinaryFormatter();
instances = (List<InstanceInformation>)binaryFormatter.Deserialize(file);
}
}
// called when the object is destroyed (you app ended)
private void OnDestroy()
{
// save the data to drive
// in the editor use the Application.streamingAssets
// in a build use Application.persistentDataPath
var folder = Application.isEditor ? Application.streamingAssets : Application.persistentDataPath;
if(!Directoy.Exists(folder)) Directory.CreateDirectory(folder);
using(var file = File.Open(Path.Combine(folder, fileName), FileMode.OpenOrCreate, FileAccess.Write, FileShare.Write))
{
var binaryFormatter = new BinaryFormatter();
binaryFormatter.Serialize(file, instances);
}
}
}
我有 3 种预制件:A、B 和 C
在设计关卡时,在播放模式下,我将它们添加并放置在关卡中。
如何创建一个 ScriptableObject 来保存对这些预制件及其转换的所有实例的引用?
具体来说,在 PlayMode 中,Scriptable 对象应动态响应 PlayMode 中预制件的位置和旋转变化。
我无法想象如何做到这一点,尽管这似乎是对脚本对象的一个很好的使用。
您可以将所需的信息存储在专用 class 中,例如
[Serializable]
public class InstanceInformation
{
public GameObject UsedPrefab;
public Vector3 Position;
public Quaternion Rotation;
public Vector3 Scale;
public InstanceInformation(GameObject usedPrefab, Transform transform)
{
UsedPrefab = usedPrefab;
Rotation = transform.rotation;
Position = transform.position;
Scale = transform.localScale;
}
public void UpdateValues(Transform transform)
{
Rotation = transform.rotation;
Position = transform.position;
Scale = transform.localScale;
}
}
并且在您的 ScriptableObject 中有一个
[CreateAssetMenu]
public class LevelData : ScriptableObject
{
public List<InstanceInformation> instances = new List<InstanceInformation>();
}
然后每次实例化一个预制件时,您也会在其中创建一个相应的条目 instances
。
所以稍后在您的管理器脚本中实例化您所做的事情,例如
// Reference this via the Inspector
public LevelData scriptable;
// For keeping a link for the currently Instantiated stuff
// so everytime you manipulate them later on you can update the according data entry
private Dictionary<GameObject, InstanceInformation> objectToInstanceInformation = new Dictionary<GameObject, InstanceInformation>();
...
var obj = Instantiate(aPrefab, aPosition, aRotation);
var instanceInfo = new InstanceInfo(aPrefab, obj.transform);
// Add the reference to the ScriptableObject list
scriptable.instances.Add(instanceInfo);
// And also keep track of the reference linked to the actual instance
objectToInstanceInformation.Add(obj, instanceInfo);
现在您可以重复或在某个时刻调用
public void SaveInstanceInformations()
{
foreach(var kvp in objectToInstanceInformation)
{
var obj = kvp.key;
var instanceInfo = kvp.value;
instanceInfo.UpdateValues(obj.transform);
}
}
因为 InstanceInformation
是一个 class,因此 reference-type 自动更改此处的值也会更改 ScriptableObject 中 instances
中的相应条目!
所以稍后当你想加载状态时你可以简单地做
foreach (var instance in scriptable.instances)
{
var obj = Instantiate(instance.UsedPrefab, instance.Position, instance.Rotation);
obj.transform.localScale = instance.Scale;
objectToInstanceInformation.Add(obj, instance);
}
最后,为了能够持久存储此数据,您可以例如使用 BinaryFormatter
喜欢
[CreateAssetMenu]
public class LevelData : ScriptableObject
{
// here you can set a filename the data shall be stored to
[SerializeField] private string fileName = "level.dat";
public List<InstanceInformation> instances = new List<InstanceInformation>();
// called when the object is loaded
private void OnEnable()
{
// try to load the data from drive
// in the editor use the Application.streamingAssetsPath
// in a build use Application.persistentDataPath
var folder = Application.isEditor ? Application.streamingAssetsPath : Application.persistentDataPath;
if(!File.Exists(Path.Combine(folder, fileName)))
{
// on the first run the folder and file will not exist
// in the editor do nothing (the file simply doesn't exist yet)
// in a build copy the content from the streaming assets folder to the persistent data path (if exists)
var fallbackFolder = Application.streamingAssetsPath;
if(!Driectoy.Exists(fallbackFolder)) return;
if(!File.Exists(Path.Combine(fallbackFolder, fileName))) return;
// copy fallback file to persistent file
File.Copy(Path.Combine(fallbackFolder, FileName), Path.Combine(folder, fileName));
}
// load the list
using(var file = File.Open(Path.Combine(folder, fileName), FileMode.Open, FileAccess.Read, FileShare.Read))
{
var binaryFormatter = new BinaryFormatter();
instances = (List<InstanceInformation>)binaryFormatter.Deserialize(file);
}
}
// called when the object is destroyed (you app ended)
private void OnDestroy()
{
// save the data to drive
// in the editor use the Application.streamingAssets
// in a build use Application.persistentDataPath
var folder = Application.isEditor ? Application.streamingAssets : Application.persistentDataPath;
if(!Directoy.Exists(folder)) Directory.CreateDirectory(folder);
using(var file = File.Open(Path.Combine(folder, fileName), FileMode.OpenOrCreate, FileAccess.Write, FileShare.Write))
{
var binaryFormatter = new BinaryFormatter();
binaryFormatter.Serialize(file, instances);
}
}
}