在Tap上更改ListView项的RowDefinition

时间:2019-02-23 00:52:34

标签: c# xaml listview xamarin

我要实现的是一个消息聊天气泡,在其中单击控件/消息,消息将展开,在下面显示日期和可见/已发送状态的详细信息。我确实有一个用于发送方和接收方的不同控件的DataTemplate选择器。

我的问题是更改ListView中消息的高度。我尝试将RowDefinition绑定到我的Message类中的Height变量(该类包含有关消息的信息) )。尽管高度已更新,但并未反映在ListView上。我已经在互联网上搜寻了现有的聊天UI模板,但我认为其中大多数都是有偿的。因此,我正在尝试遵循Change WPF DataTemplate for ListBox item if selected。但是对于Xamarin,没有ListBoxItem,因为只有ListView。

请注意,我正在使用Android和iOS。跨平台解决此问题的示例将不胜感激。下面是我的代码的一部分。

datatemplate.cs

class MessageTemplateSelector : DataTemplateSelector
{
    public MessageTemplateSelector()
    {
        ReceiverDataTemplate = new DataTemplate(typeof(MessageReceiver));
        SenderDataTemplate = new DataTemplate(typeof(MessageSender));
    }

    protected override DataTemplate OnSelectTemplate(object item, BindableObject container)
    {
        var message = item as Message;
        if (message == null)
            return null;
        return message.isSender ? ReceiverDataTemplate : SenderDataTemplate;
    }

    private readonly DataTemplate ReceiverDataTemplate;
    private readonly DataTemplate SenderDataTemplate;
}

MessageSender.xaml

<?xml version="1.0" encoding="UTF-8"?>
<ViewCell xmlns="http://xamarin.com/schemas/2014/forms" 
    xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
    x:Class="Project.layout.MessageSender">
    <ViewCell.View>
        <Grid HorizontalOptions="EndAndExpand">
            <Grid.RowDefinitions>
                <RowDefinition Height="{Binding Path=Height}"/>
                <RowDefinition Height="*" />
                <RowDefinition Height="{Binding Path=Height}" />
            </Grid.RowDefinitions>
            <Grid.ColumnDefinitions>
                <ColumnDefinition Width="75" />
                <ColumnDefinition Width="Auto" />
                <ColumnDefinition Width="15" />
            </Grid.ColumnDefinitions>
            <Label Text="{Binding Path=timestamp}" Grid.Column="0" Grid.Row="0" Grid.ColumnSpan="3" HorizontalTextAlignment="Center" HorizontalOptions="Center" VerticalOptions="Center" IsVisible="{Binding Path=Selected}"/>
            <Frame Padding="0" CornerRadius="20" Grid.Column="1" Grid.Row="1" HorizontalOptions="EndAndExpand" >
                <Grid BackgroundColor="White" VerticalOptions="FillAndExpand">
                    <Grid.ColumnDefinitions>
                        <ColumnDefinition Width="*" />
                    </Grid.ColumnDefinitions>
                    <Grid.RowDefinitions>
                        <RowDefinition Height="*" />
                    </Grid.RowDefinitions>
                    <Label Text="{Binding Path=text}" VerticalOptions="FillAndExpand" Margin="15,10"/>
                </Grid>
            </Frame>
            <Label Text="Seen" Grid.Column="1" Grid.Row="2" HorizontalOptions="EndAndExpand" IsVisible="{Binding Path=Selected}"/>
        </Grid>
  </ViewCell.View>
</ViewCell>

Message.cs

class Message
{
        public bool isSender { get; set; }
        public sbyte status { get; set; }
        public string text { get; set; }
        public string timestamp { get; set; }
        public Message(bool isSender, sbyte status, string text, string timestamp)
        {
            this.isSender = isSender;
            this.status = status;
            this.text = text;
            this.timestamp = timestamp;
        }
        public sbyte height = 0;
        public sbyte Height { get { return height; }  set { height = value; } }
        bool selected = false;
        public bool Selected
        {
            get { return selected; }
            set { selected = value;if (value) { Height = 25; } else { Height = 0; } }
        }
    }

显示mainpage.xaml

<ContentPage xmlns="http://xamarin.com/schemas/2014/forms"
             xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
             xmlns:local="clr-namespace:Project.model"
             x:Class="Project.MainPage">
    <ContentPage.Resources>
        <ResourceDictionary>
            <local:MessageTemplateSelector x:Key="MessageTemplateSelector"></local:MessageTemplateSelector>
        </ResourceDictionary>
    </ContentPage.Resources>
    <StackLayout>
        <ListView x:Name="conversation"
                  ItemTemplate="{StaticResource MessageTemplateSelector}"
                    ItemsSource="{Binding Message}"
                    HasUnevenRows="True"
                    SeparatorVisibility="None"
                    RelativeLayout.HeightConstraint="{ConstraintExpression Type=RelativeToParent,Property=Height,Factor=1,Constant=0}"
                  IsPullToRefreshEnabled="true"
                  ItemTapped="Conversation_ItemTapped"
                  Refreshing="Conversation_Refreshing">
        </ListView>
</ContentPage>

MainPage.cs

private void Conversation_ItemTapped(object sender, ItemTappedEventArgs e)
{
    if (e.Item == null) return;
    Message selectedItem = (Message)e.Item;
    Log.Debug("ItemTap","Height before:" + selectedItem.Height);
    if (selectedItem.Selected) { ((ListView)sender).SelectedItem = null; selectedItem.Selected = false; }
    else { selectedItem.Selected = true; }
    Log.Debug("ItemTap", "Height after:" + selectedItem.Height);
}

这是在ListView的ItemTapped事件中显示的我的日志的屏幕截图。如您所见,高度会更新,但不会反映在ListView上。

Proof that height updates on item tap

0 个答案:

没有答案