TinkState

Uncomplicated Reactive Library

Strength: 69

class HeroStats
{
	public int Strength;
}
class HeroStatsView : MonoBehaviour
{
	[SerializeField] TMP_Text strengthLabel;

	public void Init(HeroStats stats)
	{
		strengthLabel.text = stats.Strength.ToString();
	}
}

typical approach

  • define events/messages
  • dispatch when changing
  • subscribe for observing

typical issues

  • boilerplaty
  • error-prone
  • non-unified
  • complicated
  • costly

Strength: 69 + 420

Attack: 163 + 12

TinkState

interface Observable<T>
{
	T Value { get; }
    
	IDisposable Bind(Action<T> callback);
}
class HeroStats
{
	public Observable<int> Strength;
}
class HeroStatsView : MonoBehaviour
{
	[SerializeField] TMP_Text strengthLabel;

	public void Init(HeroStats stats)
	{
		stats.Strength.Bind(strength => strengthLabel.text = strength.ToString());
	}
}
class HeroStats
{
	public Observable<int> Strength => strength;

	readonly State<int> strength;

	public HeroStats(int initialStrength)
	{
		strength = Observable.State(initialStrength);
	}

	public void BuffMe()
	{
		strength.Value *= 2;
	}
}
class HeroStats
{
	public Observable<int> Strength => strength;
    
	public readonly Observable<int> AttackDamage;

	readonly State<int> strength;

	public HeroStats(int initialStrength)
	{
		strength = Observable.State(initialStrength);
        
		AttackDamage = Observable.Auto(() => {
        	return Balance.CalculateDamage(strength.Value)
        });
	}
}
class HeroStatsView : MonoBehaviour
{
	[SerializeField] TMP_Text strengthLabel;
	[SerializeField] TMP_Text attackDamageLabel;

	public void Init(HeroStats stats)
	{
		stats.Strength.Bind(strength =>
        	strengthLabel.text = strength.ToString());
            
		stats.AttackDamage.Bind(damage =>
        	attackDamageLabel.text = damage.ToString());
	}
}
class Weapon
{
	public readonly int DamageBonus;
}

class HeroStats
{
	public Observable<int> Strength => strength;
	public readonly Observable<int> AttackDamage;

	readonly State<int> strength;
	readonly State<Weapon> currentWeapon;

	public HeroStats(int initialStrength, Weapon initialWeapon)
	{
		strength = Observable.State(initialStrength);
		currentWeapon = Observable.State(initialWeapon);

		AttackDamage = Observable.Auto(() =>
		{
			var damage = Balance.CalculateDamage(strength.Value);
			var weapon = currentWeapon.Value;
			if (weapon != null) damage += weapon.DamageBonus;
			return damage;
		});
	}
}
  • Universal
  • Declarative
  • Sensible
  • Just works :)
IDisposable Bind(Action<T> callback);

Binding lifetimes

class HeroStatsView : MonoBehaviour
{
	[SerializeField] TMP_Text attackDamageLabel;

	public void Init(HeroStats stats)
	{
		var binding = stats.AttackDamage.Bind(damage => attackDamageLabel.text = damage.ToString());
		gameObject.DisposeOnDestroy(binding);
	}
}
class HeroStatsView : MonoBehaviour
{
	[SerializeField] TMP_Text attackDamageLabel;

	public void Init(HeroStats stats)
	{
		gameObject.RunOnActive(() =>
		{
			var binding = stats.AttackDamage.Bind(damage => attackDamageLabel.text = damage.ToString());
			return binding;
		});
	}
}

Performance principles

  • No allocations on dispatch
  • No boxing
  • No unnecessary objects

 

  • Computation result caching
  • No dispatch without changes
  • Binding batching

Extras

var list = Observable.List(new [] { 1, 2, 3 });
var sum = Observable.Auto(() => list.Sum());
Debug.Log(sum.Value); // 6 
var searchString = Observable.State("");

var foundPlayers = Observable.Auto(async () => await FindPlayers(searchString.Value));

foundPlayers.Bind(result =>
{
	switch (result.Status)
	{
		case AsyncComputeStatus.Loading:
			ShowSpinner();
			break;
		case AsyncComputeStatus.Done:
			ShowResult(result.Result);
			break;
		case AsyncComputeStatus.Failed:
			LogError(result.Exception);
			break;
	}
});

Experiments

class HeroStats : Model
{
	[Observable] public int Strength { get; private set; }
	[Observable] public int AttackDamage => Balance.CalculateDamage(Strength);
}

too much magic?

THANK YOU!

TinkState

By Dan Korostelev

TinkState

  • 228