L'engine di binding in Silverlight è uno strumento molto potente e altrettanto flessibile, che ci permette un'ottima organizzazione del codice soprattutto in ottica Model-View-ViewModel, in cui possiamo avere una buona separazione delle responsabilità di ogni classe.
In molte delle nostre pubblicazioni abbiamo visto come il ViewModel rappresenti il gestore dei dati - da e verso - l'interfaccia, appunto la View. Con la proprietà Mode=TwoWay, il valore modificato dall'utente nell'interfaccia si propaga verso la proprietà in binding, nel controllo TextBox questa propagazione avviene alla perdita del focus sul controllo. Grazie all'uso della classe EventTrigger, dell'assembly System.Windows.Interactivity (di cui abbiamo già parlato anche nello script #102), possiamo realizzare un TriggerAction che scateni l'aggiornamento del binding direttamente ogni volta che viene digitato un carattere, in concomitanza dell'evento TextChanged.
Di seguito vediamo UpdateSourceBindingAction quale l'implementazione del nostro TriggerAction:
/// <summary> /// Update the source calling BindingExpression.Update for the specified property /// </summary> [Description("Update the source calling BindingExpression.Update for the specified property")] public class UpdateSourceBindingAction : TriggerAction<FrameworkElement> { public readonly static DependencyProperty PropertyProperty = DependencyProperty.Register("Property", typeof(String), typeof(UpdateSourceBindingAction), new PropertyMetadata(null, OnPropertyChanged)); private DependencyProperty property; /// <summary> /// Initializes a new instance of the <see cref="UpdateSourceBindingAction"/> class. /// </summary> public UpdateSourceBindingAction() { } /// <summary> /// Gets or sets the property to update. /// </summary> /// <value>The property.</value> public String Property { get { return this.GetValue(PropertyProperty) as String; } set { this.SetValue(PropertyProperty, value); } } /// <summary> /// Called after the action is attached to an AssociatedObject. /// </summary> protected override void OnAttached() { base.OnAttached(); SearchProperty(); } private static void OnPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e) { UpdateSourceBindingAction u = (UpdateSourceBindingAction)d; u.SearchProperty(); } private void SearchProperty() { if (this.AssociatedObject == null) { this.property = null; return; } string propertyName = this.Property + "Property"; FieldInfo field = this.AssociatedObject.GetType().GetField(propertyName, System.Reflection.BindingFlags.FlattenHierarchy | System.Reflection.BindingFlags.Public | System.Reflection.BindingFlags.Static); if (field != null) this.property = field.GetValue(null) as DependencyProperty; else this.property = null; } /// <summary> /// Invokes the specified o. /// </summary> /// <param name="o">The o.</param> protected override void Invoke(object o) { if (property == null) return; BindingExpression be = this.AssociatedObject.GetBindingExpression(this.property); if (be != null) be.UpdateSource(); } }
Il suo utilizzo è molto semplice, ecco un esempio:
<TextBox Text="{Binding Testo, Mode=TwoWay}"> <i:Interaction.Triggers> <i:EventTrigger EventName="TextChanged"> <behaviors:UpdateSourceBindingAction Property="Text" /> </i:EventTrigger> </i:Interaction.Triggers> </TextBox>
A seguito dell'evento TextChanged viene invocato il metodo Invoke della classe TriggerAction che recupera l'espressione di binding associata alla proprietà Text (descritta con la proprietà Property), di cui forza esplicitamente la propagazione del valore verso la proprietà di destinazione, definita nell'espressione di binding.
In allegato potete trovare la classe e un esempio completo.
Commenti
Per inserire un commento, devi avere un account.
Fai il login e torna a questa pagina, oppure registrati alla nostra community.
Approfondimenti
Eliminare una project wiki di Azure DevOps
Sfruttare lo stream rendering per le pagine statiche di Blazor 8
Configurare il nome della run di un workflow di GitHub in base al contesto di esecuzione
Eseguire operazioni sui blob con Azure Storage Actions
Utilizzare Tailwind CSS all'interno di React: primi componenti
Utilizzare il metodo CountBy di LINQ per semplificare raggruppamenti e i conteggi
Utilizzare la funzione EF.Parameter per forzare la parametrizzazione di una costante con Entity Framework
Referenziare un @layer più alto in CSS
Migliorare la scalabilità delle Azure Function con il Flex Consumption
Ordine e importanza per @layer in CSS
Eseguire le GitHub Actions offline
Le novità di Angular: i miglioramenti alla CLI