I’m trying to find a way to access a variable (currently set on the Track Asset) while processing the clips in the behaviour mixer, its a scriptable object that configures some texture offsets to switch between on the timeline clips.
Is there any way to access this or do I have to add the scriptable object reference on every clip?
(which seems a messy way to handle the data as its a “track” level setting) - but I can’t understand from any examples or docs a way in which I can reference this variable.
public class LipSwitcherTrack : TrackAsset
{
public LipSwitcherPhonemsDefinition definitions;
public override Playable CreateTrackMixer(PlayableGraph graph, GameObject go, int inputCount)
{
return ScriptPlayable<LipSwitcherMixerBehaviour>.Create (graph, inputCount);
}
}
and the Mixer Behaviour PlayableBehaviour
public override void ProcessFrame(Playable playable, FrameData info, object playerData)
{
m_TrackBinding = playerData as Material;
if (!m_TrackBinding)
return;
int inputCount = playable.GetInputCount ();
float greatestWeight = 0.3f; //using a float higher than 0 to avaoid any precicion issues - we only want playables at weight = 1 or at minum higher then .5 really
//every frame it processes EVERY clip on the timeline....
for (int i = 0; i < inputCount; i++)
{
float inputWeight = playable.GetInputWeight(i); //get the weight for this "blend"
if ( inputWeight > greatestWeight){
greatestWeight = inputWeight;
ScriptPlayable<LipSwitcherBehaviour> inputPlayable = (ScriptPlayable<LipSwitcherBehaviour>)playable.GetInput(i);
LipSwitcherBehaviour input = inputPlayable.GetBehaviour ();
//
// This is were I need to be able to access the scriptable object set on the TrackAsset
//
m_offset = input.offset;
if( m_lastOffset != m_offset){
m_TrackBinding.mainTextureOffset = m_offset;
m_lastOffset = m_offset;
}
}
}
}
Alternatively you can use a ‘template’ approach - make your playable behaviour serializable and have the track store a ‘template’ of it. When you call create playable, pass the template.
The advantage of this approach, is the editor supports animation of those public fields (of certain types), similar to audio track volume.
[Serializable]
public class LipSwitcherMixerBehaviour : PlayableBehaviour
{
public LipSwitcherPhonemsDefinition definitions;
public float SomeOtherTrackVariable;
public override void ProcessFrame(...)
...
}
public class LipSwitcherTrack : TrackAsset
{
public LipSwitcherMixerBehaviour template;
public override Playable CreateTrackMixer(PlayableGraph graph, GameObject go, int inputCount)
{
return ScriptPlayable<LipSwitcherMixerBehaviour>.Create (graph, inputCount, template);
}
}
@seant_unity
thank you so much, I was having a complete mental block on this one.
You’re first solution is exactly what I was looking for, its made the implementation so much more flexible, cheers!
Thi is not always a desirable solution though: having a variable fetched from the TrackAsset makes it “global” for all the clips on that Track, that’s what should be possible.
If the variable is set on the clips at creation, it means that if you change the variable on the TrackAsset, then you need to change on the clips as well, not ideal.
Also, what happens if you drag a clip to another Track of the same type? really not ideal cause you might end up with a reference on another track. :S
Absolutely - it depends on your use case, always best to do what makes sense for your timeline and project.
For my usage this is the desired behaviour as the track is used for lip syncing and needs to hold the reference to the material and the clips contain the phoneme, which define how some texture offsets get processed on the material.
Again for my project this is the desired behaviour. When dragging between tracks I want the clips to reference a completely different material setup at the track level, but still hold their phoneme data.
It’s the desirable behaviour for your project, and that’s ok.
But it’s not an answer to the question you asked in your first post and that should be clear to whoever comes across this thread; I just wanted to point out to @seant_unity that the answer provided works for you, but doesn’t address the general issue you pointed out with your initial question ^^"