我的任务是为两个人创建一个简单的 Silverlight 聊天框。我的控制必须遵守以下要求
- 可滚动
- 如果文本太长,则必须换行
- 添加新项目/消息时,必须将该项目滚动到视图中
现在我已经成功地制作了一个用户控件来满足这些要求,但是我遇到了一个可能的错误/崩溃,我一生都无法修复。我正在寻找解决该错误的方法,或者寻找一种不同的方法来创建可滚动的聊天控件。
这是我一直在使用的代码。我们将从聊天窗口的 XAML 开始
<ListBox x:Name="lbChatHistory" Grid.Row="0" Grid.Column="0" Grid.ColumnSpan="2" ScrollViewer.VerticalScrollBarVisibility="Visible" ScrollViewer.HorizontalScrollBarVisibility="Disabled" >
<ListBox.ItemTemplate>
<DataTemplate>
<Grid Background="Beige">
<Grid.ColumnDefinitions>
<ColumnDefinition Width="70"></ColumnDefinition>
<ColumnDefinition Width="Auto"></ColumnDefinition>
</Grid.ColumnDefinitions>
<TextBlock x:Name="lblPlayer" Foreground="{Binding ForeColor}" Text="{Binding Player}" Grid.Column="0"></TextBlock>
<ContentPresenter Grid.Column="1" Width="200" Content="{Binding Message}" />
</Grid>
</DataTemplate>
</ListBox.ItemTemplate>
</ListBox>
这个想法是向列表框添加一个新项目。项目(如 XAML 中所示)是一个简单的 2 列网格。一栏是用户名,一栏是消息。
现在,我添加到 ListBox 的“项目”是一个自定义类。它具有我在 XAML 中使用绑定的三个属性(Player、ForeColor 和 Message)
Player是要显示的当前用户的字符串。
ForeColor只是一种前景色偏好。它有助于区分消息之间的差异。
Message是WrapPanel。我以编程方式在每个单词的空白处打断提供的字符串。然后对于每个单词,我将一个新的TextBlock元素添加到WrapPanel
这是自定义类。
public class ChatMessage :DependencyObject, INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
public static DependencyProperty PlayerProperty = DependencyProperty.Register( "Player", typeof( string ), typeof( ChatMessage ),
new PropertyMetadata(
new PropertyChangedCallback( OnPlayerPropertyChanged ) ) );
public static DependencyProperty MessageProperty = DependencyProperty.Register( "Message", typeof( WrapPanel ), typeof( ChatMessage ),
new PropertyMetadata(
new PropertyChangedCallback( OnMessagePropertyChanged ) ) );
public static DependencyProperty ForeColorProperty = DependencyProperty.Register( "ForeColor", typeof( SolidColorBrush ), typeof( ChatMessage ),
new PropertyMetadata(
new PropertyChangedCallback( OnForeColorPropertyChanged ) ) );
private static void OnForeColorPropertyChanged( DependencyObject d, DependencyPropertyChangedEventArgs e )
{
ChatMessage c = d as ChatMessage;
c.ForeColor = ( SolidColorBrush ) e.NewValue;
}
public ChatMessage()
{
Message = new WrapPanel();
ForeColor = new SolidColorBrush( Colors.White );
}
private static void OnMessagePropertyChanged( DependencyObject d, DependencyPropertyChangedEventArgs e )
{
ChatMessage c = d as ChatMessage;
c.Message = ( WrapPanel ) e.NewValue;
}
private static void OnPlayerPropertyChanged( DependencyObject d, DependencyPropertyChangedEventArgs e )
{
ChatMessage c = d as ChatMessage;
c.Player = e.NewValue.ToString();
}
public SolidColorBrush ForeColor
{
get { return ( SolidColorBrush ) GetValue( ForeColorProperty ); }
set
{
SetValue( ForeColorProperty, value );
if(PropertyChanged != null)
PropertyChanged(this, new PropertyChangedEventArgs( "ForeColor" ));
}
}
public string Player
{
get { return ( string ) GetValue( PlayerProperty ); }
set
{
SetValue( PlayerProperty, value );
if ( PropertyChanged != null )
PropertyChanged( this, new PropertyChangedEventArgs( "Player" ) );
}
}
public WrapPanel Message
{
get { return ( WrapPanel ) GetValue( MessageProperty ); }
set
{
SetValue( MessageProperty, value );
if ( PropertyChanged != null )
PropertyChanged( this, new PropertyChangedEventArgs( "Message" ) );
}
}
}
最后,我将我的项目添加到 ListBox。这是简单的方法。它将上面的 ChatMessage 类作为参数
public void AddChatItem( ChatMessage msg )
{
lbChatHistory.Items.Add( msg );
lbChatHistory.ScrollIntoView( msg );
}
现在我已经对此进行了测试,并且一切正常。我遇到的问题是当我使用滚动条时。您可以使用侧滚动条或箭头键向下滚动,但向上滚动时 Silverlight 会崩溃。FireBug 返回带有XamlParseException的ManagedRuntimeError #4004。
我非常接近进行这种控制工作,我可以品尝到它!对我应该做什么或改变有什么想法吗?有没有比我采用的方法更好的方法?
提前致谢。
更新
我找到了使用 ScrollViewer 和 ItemsControl 而不是 ListBox 控件的替代解决方案。在大多数情况下,它是稳定的。