Creación de comportamientos de Xamarin.Forms
Xamarin.FormsLos comportamientos de se crean mediante la derivación de la clase Behavior o Behavior<T>. En este artículo se explica cómo crear y consumir comportamientos de Xamarin.Forms.
El proceso de creación de un comportamiento de Xamarin.Forms es el siguiente:
- Cree una clase que herede de la clase
Behavior
oBehavior<T>
, dondeT
sea el tipo del control al que se debe aplicar el comportamiento. - Invalide el método
OnAttachedTo
para realizar cualquier configuración necesaria. - Invalide el método
OnDetachingFrom
para realizar cualquier limpieza necesaria. - Implemente la funcionalidad básica del comportamiento.
El resultado es la estructura que se muestra en el ejemplo de código siguiente:
public class CustomBehavior : Behavior<View>
{
protected override void OnAttachedTo (View bindable)
{
base.OnAttachedTo (bindable);
// Perform setup
}
protected override void OnDetachingFrom (View bindable)
{
base.OnDetachingFrom (bindable);
// Perform clean up
}
// Behavior implementation
}
El método OnAttachedTo
se desencadena justo después de que se adjunte el comportamiento a un control. Este método recibe una referencia al control al que se adjunta, y se puede usar para registrar controladores de eventos o realizar otra configuración necesaria para admitir la funcionalidad del comportamiento. Por ejemplo, se podría suscribir a un evento en un control. Después, se implementaría la funcionalidad del comportamiento en el controlador de eventos para el evento.
El método OnDetachingFrom
se desencadena cuando se quita el comportamiento del control. Este método recibe una referencia al control al que se adjunta y se usa para realizar cualquier limpieza necesaria. Por ejemplo, podría cancelar la suscripción a un evento en un control para evitar fugas de memoria.
Después, el comportamiento se puede consumir si se adjunta a la colección Behaviors
del control adecuado.
En la aplicación de ejemplo se muestra un elemento NumericValidationBehavior
, que resalta el valor especificado por el usuario en un control Entry
en color rojo, si no es de tipo double
. El comportamiento se muestra en el ejemplo de código siguiente:
public class NumericValidationBehavior : Behavior<Entry>
{
protected override void OnAttachedTo(Entry entry)
{
entry.TextChanged += OnEntryTextChanged;
base.OnAttachedTo(entry);
}
protected override void OnDetachingFrom(Entry entry)
{
entry.TextChanged -= OnEntryTextChanged;
base.OnDetachingFrom(entry);
}
void OnEntryTextChanged(object sender, TextChangedEventArgs args)
{
double result;
bool isValid = double.TryParse (args.NewTextValue, out result);
((Entry)sender).TextColor = isValid ? Color.Default : Color.Red;
}
}
NumericValidationBehavior
se deriva de la clase Behavior<T>
, donde T
es un elemento Entry
. El método OnAttachedTo
registra un controlador de eventos para el evento TextChanged
, y el método OnDetachingFrom
anula el registro del evento TextChanged
para evitar fugas de memoria. El método OnEntryTextChanged
proporciona la funcionalidad básica del comportamiento y analiza el valor especificado por el usuario en el elemento Entry
y establece la propiedad TextColor
en rojo si el valor no es de tipo double
.
Nota
Xamarin.Forms no establece el elemento BindingContext
de un comportamiento, ya que los comportamientos se pueden compartir y aplicar a varios controles mediante estilos.
Todos los controles de Xamarin.Forms tienen una colección Behaviors
, a la que se pueden agregar uno o varios comportamientos, como se muestra en el ejemplo de código XAML siguiente:
<Entry Placeholder="Enter a System.Double">
<Entry.Behaviors>
<local:NumericValidationBehavior />
</Entry.Behaviors>
</Entry>
El control Entry
equivalente en C# se muestra en el ejemplo de código siguiente:
var entry = new Entry { Placeholder = "Enter a System.Double" };
entry.Behaviors.Add (new NumericValidationBehavior ());
En tiempo de ejecución, el comportamiento responderá a la interacción con el control, en función de la implementación del comportamiento. En las capturas de pantalla siguientes se muestra la respuesta del comportamiento a entradas no válidas:
Nota
Los comportamientos se escriben para un tipo de control específico (o una superclase que se puede aplicar a muchos controles), y solo se deben agregar a un control compatible. El intento de adjuntar un comportamiento a un control incompatible iniciará una excepción.
Los comportamientos también se pueden consumir mediante un estilo explícito o implícito. Pero no se puede crear un estilo que establece la propiedad Behaviors
de un control porque la propiedad es de solo lectura. La solución consiste en agregar una propiedad adjunta a la clase de comportamiento que controle la adición y eliminación del comportamiento. El proceso es el siguiente:
- Se agrega una propiedad adjunta a la clase de comportamiento que se usará para controlar la adición o eliminación del comportamiento del control al que se va a conectar el comportamiento. Se asegura que la propiedad adjunta registra un delegado
propertyChanged
que se ejecutará cuando cambie el valor de la propiedad. - Se crea un captador y establecedor
static
para la propiedad adjunta. - Se implementa la lógica en el delegado
propertyChanged
para agregar y quitar el comportamiento.
En el ejemplo de código siguiente se muestra una propiedad adjunta que controla la adición y eliminación de NumericValidationBehavior
:
public class NumericValidationBehavior : Behavior<Entry>
{
public static readonly BindableProperty AttachBehaviorProperty =
BindableProperty.CreateAttached ("AttachBehavior", typeof(bool), typeof(NumericValidationBehavior), false, propertyChanged: OnAttachBehaviorChanged);
public static bool GetAttachBehavior (BindableObject view)
{
return (bool)view.GetValue (AttachBehaviorProperty);
}
public static void SetAttachBehavior (BindableObject view, bool value)
{
view.SetValue (AttachBehaviorProperty, value);
}
static void OnAttachBehaviorChanged (BindableObject view, object oldValue, object newValue)
{
var entry = view as Entry;
if (entry == null) {
return;
}
bool attachBehavior = (bool)newValue;
if (attachBehavior) {
entry.Behaviors.Add (new NumericValidationBehavior ());
} else {
var toRemove = entry.Behaviors.FirstOrDefault (b => b is NumericValidationBehavior);
if (toRemove != null) {
entry.Behaviors.Remove (toRemove);
}
}
}
...
}
La clase NumericValidationBehavior
contiene una propiedad asociada denominada AttachBehavior
con un captador y establecedor de static
, que controla la adición o eliminación del comportamiento del control al que se va a asociar. Esta propiedad adjunta registra el método OnAttachBehaviorChanged
que se ejecutará cuando cambie el valor de la propiedad. Este método agrega o quita el comportamiento del control, en función del valor de la propiedad adjunta AttachBehavior
.
En el ejemplo de código siguiente se muestra un estilo explícito para el elemento NumericValidationBehavior
que usa la propiedad adjunta AttachBehavior
y que se puede aplicar a controles Entry
:
<Style x:Key="NumericValidationStyle" TargetType="Entry">
<Style.Setters>
<Setter Property="local:NumericValidationBehavior.AttachBehavior" Value="true" />
</Style.Setters>
</Style>
Style
se puede aplicar a un control Entry
si se establece su propiedad Style
en la instancia de Style
mediante la extensión de marcado StaticResource
, como se muestra en el ejemplo de código siguiente:
<Entry Placeholder="Enter a System.Double" Style="{StaticResource NumericValidationStyle}">
Para obtener más información sobre los estilos, consulta Estilos.
Nota
Aunque se pueden agregar propiedades enlazables a un comportamiento que se establece o se consulta en XAML, si se crean comportamientos que tienen estado, no se deberían compartir entre los controles en un elemento Style
de un objeto ResourceDictionary
.
El método OnDetachingFrom
se desencadena cuando se quita un comportamiento de un control, y se usa para realizar cualquier limpieza necesaria como cancelar la suscripción de un evento para evitar una fuga de memoria. Pero los comportamientos no se quitan de forma implícita de los controles a menos que un método Remove
o Clear
modifique la colección Behaviors
del control. En el ejemplo de código siguiente se muestra cómo quitar un comportamiento específico de la colección Behaviors
de un control:
var toRemove = entry.Behaviors.FirstOrDefault (b => b is NumericValidationBehavior);
if (toRemove != null) {
entry.Behaviors.Remove (toRemove);
}
Como alternativa, se puede borrar la colección Behaviors
del control, como se muestra en el ejemplo de código siguiente:
entry.Behaviors.Clear();
Además, tenga en cuenta que los comportamientos no se quitan de forma implícita de los controles cuando se extraen páginas de la pila de navegación. En su lugar, se deben quitar explícitamente antes de que las páginas queden fuera del ámbito.
En este artículo se explica cómo crear y consumir comportamientos de Xamarin.Forms. Los comportamientos de Xamarin.Forms se crean mediante la derivación de la clase Behavior
o Behavior<T>
.