gpt4 book ai didi

c# - 仅附加在第一个用户控件实例上的附加属性

转载 作者:行者123 更新时间:2023-12-03 10:55:38 25 4
gpt4 key购买 nike

在 mvvm 工作区(客户 View )上的 Josh Smith 示例之后,我有一个主窗口和一个主窗口 View 模型,其中包含“ChatTabViewModel”的 ObservableCollection:

internal class FriendsListViewModel : ObservableObject
{
#region bound properties
private ICollectionView viewfriends;
private ObservableCollection<ChatTabViewModel> _chatTab;
...
#endregion
}

我在 xaml 中有一个专门用于此集合的区域:
<ContentControl Grid.Column="0" Grid.Row="0" Grid.RowSpan="2" Content="{Binding Path=ChatTabs}" ContentTemplate="{StaticResource ChatTabsTemplate}" />

在我的资源字典中:
<DataTemplate DataType="{x:Type vm:ChatTabViewModel}">
<View:ChatTabView />
</DataTemplate>

<DataTemplate x:Key="ClosableTabItemTemplate">
<DockPanel>
<Button
Command="{Binding Path=CloseCommand}"
Content="X"
Cursor="Hand"
DockPanel.Dock="Right"
Focusable="False"
FontFamily="Courier"
FontSize="9"
FontWeight="Bold"
Margin="0,1,0,0"
Padding="0"
VerticalContentAlignment="Bottom"
Width="16" Height="16"
/>
<ContentPresenter
Content="{Binding Path=Caption, Mode=OneWay}"
VerticalAlignment="Center">
</ContentPresenter>
</DockPanel>
</DataTemplate>

<DataTemplate x:Key="ChatTabsTemplate">
<TabControl
IsSynchronizedWithCurrentItem="True"
ItemsSource="{Binding}"
ItemTemplate="{StaticResource ClosableTabItemTemplate}"
Margin="4"/>
</DataTemplate>

在用户事件中,我在我的集​​合中添加了一个新的 ChattabViewModel,并且与之相关的 View 出现在主窗口中。

但是当我尝试在 ChattabView 的滚动条上添加附加属性时,该属性将仅附加在第一个 ChattabViewModel 实例上,其他选项卡不会绑定(bind)到附加属性。这是 ChattabView XAML:
 <ScrollViewer VerticalScrollBarVisibility="Auto" Grid.Row="0">
<ItemsControl ItemsSource="{Binding Messages}" View:ItemsControlBehavior.ScrollOnNewItem="True">
<ItemsControl.ItemTemplate>
<DataTemplate>
<TextBox IsReadOnly="True" TextWrapping="Wrap" Text="{Binding Path=DataContext, RelativeSource={RelativeSource Self}}" />
</DataTemplate>
</ItemsControl.ItemTemplate>
</ItemsControl>
</ScrollViewer>

和附加属性的代码:
namespace GtalkOntre.View
{
/// <summary>
/// Util class to scroll down when a new message is added.
/// </summary>
/// <remarks>attached property called ScrollOnNewItem that when set to true hooks into the INotifyCollectionChanged events of the itemscontrol items source and upon detecting a new item, scrolls the scrollbar to it.</remarks>
public class ItemsControlBehavior
{
static Dictionary<ItemsControl, Capture> Associations = new Dictionary<ItemsControl, Capture>();

public static bool GetScrollOnNewItem(DependencyObject obj)
{
return (bool)obj.GetValue(ScrollOnNewItemProperty);
}

public static void SetScrollOnNewItem(DependencyObject obj, bool value)
{
obj.SetValue(ScrollOnNewItemProperty, value);
}

public static DependencyProperty ScrollOnNewItemProperty =
DependencyProperty .RegisterAttached(
"ScrollOnNewItem",
typeof(bool),
typeof(ItemsControlBehavior),
new UIPropertyMetadata(false, OnScrollOnNewItemChanged));

public static void OnScrollOnNewItemChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var mycontrol = d as ItemsControl;
if (mycontrol == null) return;
bool newValue = (bool)e.NewValue;
if (newValue)
{
mycontrol.Loaded += MyControl_Loaded;
mycontrol.Unloaded += MyControl_Unloaded;
}
else
{
mycontrol.Loaded -= MyControl_Loaded;
mycontrol.Unloaded -= MyControl_Unloaded;
if (Associations.ContainsKey(mycontrol))
Associations[mycontrol].Dispose();
}
}

static void MyControl_Unloaded(object sender, RoutedEventArgs e)
{
var mycontrol = (ItemsControl)sender;
Associations[mycontrol].Dispose();
mycontrol.Unloaded -= MyControl_Unloaded;
}

static void MyControl_Loaded(object sender, RoutedEventArgs e)
{
var mycontrol = (ItemsControl)sender;
var incc = mycontrol.Items as INotifyCollectionChanged;
if (incc == null) return;
mycontrol.Loaded -= MyControl_Loaded;
Associations[mycontrol] = new Capture(mycontrol);
}

class Capture : IDisposable
{
public ItemsControl mycontrol { get; set; }
public INotifyCollectionChanged incc { get; set; }

public Capture(ItemsControl mycontrol)
{
this.mycontrol = mycontrol;
incc = mycontrol.ItemsSource as INotifyCollectionChanged;
incc.CollectionChanged +=incc_CollectionChanged;
}

void incc_CollectionChanged(object sender, NotifyCollectionChangedEventArgs e)
{
if (e.Action == NotifyCollectionChangedAction.Add)
{
ScrollViewer sv = mycontrol.Parent as ScrollViewer;
sv.ScrollToBottom();
}
}

public void Dispose()
{
incc.CollectionChanged -= incc_CollectionChanged;
}
}
}
}

那么为什么附加属性只绑定(bind)一次,在 chattabviewmodel 集合的第一次“chattabview”出现时?因此,仅在第一个 chattabviewmodel 上工作。
当我全部关闭它们时,附加属性将在 chattabviewmodel 的最后一个实例上解除绑定(bind),并且当我添加新的第一个 chattabviewmodel 时,该属性将正确绑定(bind)。因此它仅在主窗口 View 模型的“chattabviewmodel”集合的第一个实例和最后一个实例上触发。

经过一周的搜索,我现在有点绝望......

到目前为止,我的假设是:问题可能与我在字典资源中将 View 设置为 View 模型的方式有关。 View 可能是共享的,并且只有第一个滚动条可能会使用react。我尝试添加 x:Shared = false DataTemplate 标记上的属性,但它没有改变任何东西。

最佳答案

你确定你的ChatTabView有不同的实例吗?被创造?

我相信 WPF 的 TabControl如果相同,则重新使用现有模板,而不是创建新模板,并简单地替换 DataContext在它后面。

所以它只会创建您的 ChatTabView 的一份副本和切换标签正在取代 DataContext背后ChatTabView到集合中的不同项目。

关于c# - 仅附加在第一个用户控件实例上的附加属性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9159411/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com