2009-03-02 11 views
39

NOTA Ho chiesto la relativa domanda: How to combine DataTrigger and EventTrigger?WPF - Come combinare DataTrigger e Trigger?

Ho una scatola elenco contenente diversi elementi. La classe dell'oggetto implementa INotifyPropertyChanged e ha una proprietà IsAvailable. Io uso quella proprietà per indicare opzioni non disponibili nella lista usando un colore diverso.

Tuttavia, se un elemento selezionato non è disponibile, il colore di primo piano dovrebbe essere rosso.

<ListBox> 
    <ListBox.Resources> 
    <DataTemplate DataType="{x:Type local:InstitutionViewModel}"> 
     <TextBlock Name="Name" Text="{Binding Name}"/> 
     <DataTemplate.Triggers> 
     <DataTrigger Binding="{Binding IsAvailable}" Value="False"> 
      <Setter TargetName="Name" Property="Foreground" Value="#888"/> 
     </DataTrigger> 
     </DataTemplate.Triggers> 
    </DataTemplate> 
    </ListBox.Resources> 
</ListBox> 

Uso il trigger di dati sopra riportato per escludere elementi non disponibili.

Il problema che sto affrontando è che il fatto che l'elemento sia selezionato non ha nulla a che fare con i dati sottostanti a cui è associato il modello. Quello che voglio veramente è una sorta di multi-trigger che supporta sia un normale Trigger su una proprietà di dipendenza (ListBoxItem.IsSelected) insieme a un DataTrigger sull'elemento di dati associato.

Questo può essere fatto senza introdurre il concetto di selezione nel mio modello di vista?

Per chiunque si chieda il motivo per cui non disattivo gli articoli non disponibili, è necessario specificare che è possibile che siano disponibili opzioni non disponibili. In effetti ci sono alcune caselle di elenco e la selezione in uno degli effetti disponibili negli altri. Non posso disabilitare gli elementi in quanto l'utente non sarebbe in grado di cambiare idea o esplorare combinazioni diverse se gli elementi fossero disabilitati in base alle selezioni precedenti.

risposta

51

Per chiunque abbia problemi con questo problema, ho trovato una soluzione che funziona per me. Certo, sono ancora interessato a vedere altre risposte interessanti.

Ecco quello che ho fatto:

<MultiDataTrigger> 
    <MultiDataTrigger.Conditions> 
    <Condition Binding="{Binding 
     RelativeSource={ 
     RelativeSource Mode=FindAncestor, AncestorType={x:Type ListBoxItem}}, 
     Path=IsSelected}" Value="True"/> 
    <Condition Binding="{Binding IsAvailable}" Value="False"/> 
    </MultiDataTrigger.Conditions> 
    <Setter TargetName="Name" Property="Foreground" Value="#F00"/> 
</MultiDataTrigger> 

Non c'è niente di speciale in questo essere un multi grilletto però. Se si voleva solo per lo stile l'elemento selezionato in modo diverso nel vostro modello di dati, è possibile utilizzare:

<DataTrigger Binding="{Binding 
    RelativeSource={ 
    RelativeSource Mode=FindAncestor, AncestorType={x:Type ListBoxItem}}, 
    Path=IsSelected}" Value="True"> 
    <Setter TargetName="Name" Property="Foreground" Value="#888"/> 
</DataTrigger> 
+2

che è esattamente quello che stavo raccomandare. Per quanto ne so è la soluzione migliore. –

+0

basta colpire questo stesso problema. Eccellente punto di vista, dimenticato il legame relativo alla fonte. – Stimul8d

+0

eccellente! grazie per questo! – Fredrik

9

Per usarlo con la modalità di rilegatura DataGridRow modifica Self:

Binding="{Binding RelativeSource={RelativeSource Mode=Self}, Path=...