Skip to content

Latest commit

 

History

History
385 lines (321 loc) · 14.6 KB

README_CN.md

File metadata and controls

385 lines (321 loc) · 14.6 KB

UBind [Preview]

UBind 是一个适用于 Unity 的数值绑定组件,用于快速实现UI和逻辑数据的关联绑定。

license openupm Unity: 2019.4.3f1 .NET 4.x topLanguage size last 996.icu

issue PRs Welcome Updates

[English Documents Available]

官方交流QQ群:1070645638


1. 特性

  • 支持双向绑定
  • 支持多数据源和多目标对象
  • 支持任意组件的任意属性和字段绑定
  • 支持数据源和目标数据类型不同时的自动类型转换
  • 提供大量内置组件的常用属性绑定器
  • 任意运行时对象的属性和字段绑定
  • 提供数据容器以按组划分数据使用域
  • 可扩展自定义数据绑定器、自定义类型转换器

2. 使用方法

2.1. 组件方式

2.1.1. 常用组件绑定器

可以使用内置的绑定器组件直接将常见UI组件的属性设置成数据源或者数据目标:

Text Binder

2.1.2. 属性绑定器

如果需要操作自定义组件,或者未内置提供专用绑定器的组件,则可以使用更为通用的 PropertyBinder,简单设置即可绑定任意组件的任意属性字段:

Property Binder

2.1.3. 类型绑定器

如果需要将一个数据类绑定到对应的UI,则只需要使用 TypeBinder,指定数据类的程序集和类名,然后将对应属性字段与UI依次绑定,然后在提供数据的逻辑代码中,编码绑定数据源即可:

public class UBindSamplePlayerData
{
    public string Name;
    public int Exp;
    public float Stat;
    public float PlayTime;
}

Type Binder

public class UBindSampleGameManager
{
    public UBindSamplePlayerData Player;

    public void Awake()
    {
        Player = new UBindSamplePlayerData() {Name = "Player",};
        UBind.BindSource("PlayerData", Player);
    }
}

2.2. 代码方式

2.2.1. MonoBehaviour 自动绑定

通过继承 BindableMonoBehaviour 获得自动处理属性和字段绑定、解绑的能力。

使用 BindValue Attribute 来标记需要绑定基础类型数据:

public class ExampleMonoBehaviour : BindableMonoBehaviour
{
	[BindValueSource("ValueKey")]
	public string ValueSource;

	[BindValueTarget("ValueKey")]
	public string ValueTarget;
}

使用 BindType Attribute 来标记需要绑定类和结构体数据:

public class ExampleData
{
	public string Value;
}

public class ExanokeMonoBehaviour : BindableMonoBehaviour
{
	[BindTypeSource("TypeKey")]
	public ExampleData DataSource;

	[BindTypeTarget("TypeKey")]
	public ExampleData DataTarget;
}

2.2.2. MonoBehaviour 手动绑定

对于无法使用继承的自定义 Monobehaviour 对象,可以在 OnEnable / OnDisable方法中添加如下代码,手动调用 BindMap 的绑定和解绑接口,可以获得和自动绑定一样的效果:

public class ExanokeMonoBehaviour : MonoBehaviour
{
	[BindValueSource("ValueKey")]
	public string ValueSource;

	[BindValueTarget("ValueKey")]
	public string ValueTarget;

	public void OnEnable()
	{
		UBind.RegisterMap(this);
	}

	public void OnDisable()
	{
		UBind.DeRegisterMap(this);
	}
}

2.2.3. 手动绑定任意对象

通过调用 UBind 的多种重载接口,可以实现对运行时的数值、属性、字段、类对象、结构体对象进行绑定,如果需要在特定时间进行绑定和解绑操作,则需要自行缓存返回的 DataBinder 对象:

public class ExampleData
{
	public string Value;
}

public class ExampleClass
{
	public string ValueSource;
	public string ValueTarget;

	public ExampleData TypeSource;
	public ExampleData TypeTarget;

	private DataBinder _runtimeValueSourceBinder;
	private DataBinder _runtimeValueTargetBinder;

	private DataBinder _runtimeTypeSourceBinder;
	private DataBinder _runtimeTypeTargetBinder;

	public void BindTest()
	{
		_runtimeValueSourceBinder = UBind.BindSource<string>("ValueKey", () => ValueSource);
		_runtimeValueTargetBinder = UBind.BindTarget<string>("ValueKey", value => ValueTarget = value);

		_runtimeTypeSourceBinder = UBind.BindSource("TypeKey", TypeSource);
		_runtimeTypeTargetBinder = UBind.BindTarget("TypeKey", TypeTarget);
	}

	public void UnBindTest()
	{
		_runtimeValueSourceBinder.UnBind();
		_runtimeValueTargetBinder.UnBind();

		_runtimeTypeSourceBinder.UnBind();
		_runtimeTypeTargetBinder.UnBind();
	}
}

需要注意的是,在以上的例子中,数据源和数据目标各生成了一个绑定器,这是默认并且推荐的使用方式,以便于支持多数据源、多数据目标。但是数值类的绑定 (RuntimeValueBinder<T>) 也可以使用下面的方式简化调用,会同时返回数据源绑定器和数据目标绑定器:

public class ExampleClass
{
	public string Source;
	public string Target;

	private DataBinder _sourceBinder;
	private DataBinder _targetBinder;

	public void BindTest()
	{
		(_sourceBinder, _targetBinder) = UBind.Bind<string>("Key", () => Source, value => Target = value);
	}

	public void UnBindTest()
	{
		_sourceBinder.UnBind();
		_targetBinder.UnBind();
	}
}

3. 工作模式

3.1. 被动回调模式

OnValueChangedTrigger -> DataBinder(Source) --> DataConverter --> DataBinder(Target)

默认推荐使用的模式,此时 DataBinderNeedUpdate 属性为 false,数据由数据源提供的数值变化回调触发刷新,通知到数据源绑定器,广播给所有的数据目标绑定器,再经由数据转换器转换成目标数据类熊。

3.2. 主动更新模式

BindUpdater -> DataBinder(Source) --> DataConverter --> DataBinder(Target)

当数据源不具备主动触发数据变化的能力时,框架提供统一的更新周期来主动检测数值变化,但这个模式无法避免的会造成性能损耗,应尽量减少使用,比较适合快速原型阶段。在这个模式下,DataBinderNeedUpdate 属性为 true,且需要实现 AddListener()RemoveListener() 方法实现具体的回调注册和注销。框架在 BindUpdater 组件中统一按周期触发所有数据源的 UpdateSource() 方法以实现数据变化检测和数据更新广播。


4. 内建类型

4.1. Attribute

4.1.1. Bind Value Attribute

用于在拥有处理绑定关系能力的类中标记需要绑定的属性和字段,只推荐绑定常见基础数据类型。被标记对象会动态创建 RuntimeValueBinder 进行处理。

4.1.2. Bind Type Attribute

BindValueAttribute 的不同,用于标记自定义类和结构体类型对象,会动态创建 RuntimeTypeBinder 进行处理。

4.2. Data Container

数据容器,用于维护一组 Data Binder,可以包含多个数据源和数据目标点。每个数据容器相互独立。

4.3. Data Binder

数据绑定器,用于将数据与具体的对象和其属性进行绑定,可以用于接收和广播数据。

属性 描述
Target 数据目标对象。
Context 数据容器的 Key 值,默认为 Default,无特殊需求可保持默认不设置。
Key 目标数据的 Key 值,是单个数据容器内数据的唯一标识。
Direction 数据的传递方向,Source Target 标识当前绑定器作为数据接收者。
UpdateType 对于未提供数据变化回调的目标对象,数据采用 Update 模式检测变化是否发生,此时需要指定 Update 的时机。

4.4. Data Converter

当数据源和数据目标的数据类型不同时,会使用与 (sourceType, targetType) 对应的数据转换器尝试转换。

默认的转换器 CommonConverter 的实现如下,可以通过修改 DataConverter.Default 来替换:

Convert.ChangeType(object data, Type type);

可以按需要使用以下接口提前注册特定类型的自定义转换器:

UBind.RegisterConverter(Type sourceType, Type targetType, DataConverter dataConverter);

4.5. Component Binder

数据绑定器的 Unity Component 版本,需要与一个具体的绑定器实现泛型绑定,用于实现组件数据的绑定。

4.6. Bind Updater

基于 MonoBehaviour 生命周期实现,用于统一维护需要主动更新数据的 DataBinder 的更新周期。

4.7. Bind Map

用于缓存类中所有被 BindAttribute 所标记的字段和属性信息。


5. 内建组件

组件 绑定对象类型 默认绑定属性 绑定属性类型
Text Binder Text text string
Text FontSize Binder Text fontSize int
Text Format Value Binder Text text float
InputField Binder InputField text string
Slider Binder Slider value float
Scrollbar Binder Scrollbar value float
Dropdown Binder Dropdown value int
Dropdown List Binder Dropdown options List<Dropdown.OptionData>
CanvasGroup Binder CanvasGroup alpha float
Toggle Binder Toggle value bool
Color Binder Graphic color Color
Image Binder Image sprite Sprite
Image FillAmount Binder Image fillAmount float
RawImage Binder RawImage texture Texture
Property Binder Component
Type Binder Component

6. 自定义扩展

6.1. Custom Data Binder

通过继承 DataBinder 实现一个特定对象和属性的运行时数据绑定器:

public class RuntimeImageFillAmountBinder : DataBinder<Image, float>
{
	public override float Value
	{
		get => Target.fillAmount;
		set => Target.fillAmount = value;
	}
}

6.2. Custom Component Binder

如果需要用作组件,则需要额外实现对应的 ComponentBinder

[AddComponentMenu("Data Binding/Image FillAmount Binder")]
public class ImageFillAmountBinder : ComponentUpdateBinder<Image, float, RuntimeImageFillAmountBinder>
{
}

6.3. Custom Binder Editor

如果组件需要保持基础组件样式并且扩展自定义属性和样式,则需要实现对应的 ComponentBinderEditor,与绑定器组件和运行时绑定器泛型关联:

#if UNITY_EDITOR
using UnityEditor;

[CustomEditor(typeof(ImageFillAmountBinder)), UnityEditor.CanEditMultipleObjects]
public class ImageFillAmountBinderEditor : ComponentBinderEditor<Image, float, RuntimeImageFillAmountBinder>
{
}
#endif

6.4. Custom Data Converter

如果需要在传递不同数据类型,并且数据不是 Convert.ChangeType() 可以处理的,则需要实现该类型转换的自定义数据转换器:

public class CustomDataConverter : DataConverter
{
	public object To(object data, Type targetType)
	{
		object result;
		// To do something...
		return result;
	}
}

需要使用这些绑定器之前,比如游戏初始化时,注册自定的绑定器:

var sourcetType = sourceData.GetType();
var targetType = targetData.GetType();
UBind.RegisterConverter((sourceType, targetType), new CustomDataConverter());

7. 注意事项

  • OnValueChanged 仅数据源可用。
  • 所有参数必须在编辑器模式配置,运行时修改参数无效。
  • TypeBinder 暂时仅支持单一层级属性的简单数据类。
  • 尽管组件支持不同数据类型的自动转换,但任然建议直接传递相同的数据类型以优化性能。
  • 多个数据源之间的数据不会被同步,如果数据源之间也需要同步需要将 Direction 设置成 Both。
  • PropertyBinder 和 TypeBinder 依赖反射实现,但由于某些原生组件的属性封装,直接反射修改字段并不能刷新显示,可能需要操作对应的属性,或者定制专用绑定器。

8. 扩展 -- TextMeshPro

  • TMP Text Binder
  • TMP Text Format Value Binder