gpt4 book ai didi

Xamarin:在后台更新 ObservableCollection

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

我有一个基于由 ObservableCollection 填充的 Listview 并绑定(bind)到公开为 List 的 SQLite 模型的 Xamarin.Forms 应用程序。

现在我很难弄清楚如何通过网络更新数据。

整个过程应该在后台运行。如果一切都按预期运行,则更新过程将基于 async ... await 任务,并且新项目将在被拉入时一一出现在屏幕上。

谁能指导我如何布置我的应用程序以及如何实现这样的后台更新任务?

笔记:

更有经验的同事警告我,这样的概念根本无法在 Xamarin 中完成,因为他们说 ObservableCollection“不支持通过后台任务进行更新”。我对此进行了一些研究,发现确实有一些迹象表明这可能是真的,但这些信息早已过时(从 2008 年开始),从那时起情况很可能发生了变化。

出于性能原因,我不能简单地拉入完整列表并丢弃现有列表,但我需要实现基于记录的更新,逐个查看项目。为完成此记录,记录具有唯一的 ID,并带有时间戳。我的应用程序知道它最后一次看到 Web 服务的时间,并且可以查询从那时起发生变化的所有项目。我已经有一个 REST 服务从后端提取更改的项目数据作为列表,但无法弄清楚如何刷新 ObservableCollection 和底层数据库。

最佳答案

我对 ListView 的更改进行了所有更新。在这里,我在 ListView 中有一个按钮,单击该按钮会更新一个属性,该属性通过将其保存到 sql 数据库中而持久存在。它假设您已经设置了数据库。

数据库 :
如果存在则更新项目或保存新项目的功能。这是一个任务,因此可以异步调用。

   public Task<int> SaveItemAsync(Item item)
{
if(item.ItemId != 0)
{
return database.UpdateAsync(item);
}
else
{
return database.InsertAsync(itme);
}
}

Xaml
  • 绑定(bind)到从项目数据库创建的 Observable 集合的 ListView 。
  • GestureRecognizers 在图像上设置并绑定(bind)到 ViewModel 中的 tapCommand - Xaml 背后的代码定义了绑定(bind)上下文

  • 背后的代码
     ItemViewModel viewModel;
    public MessagePage ()
    {
    InitializeComponent ();
    viewModel = new ItemViewModel();
    BindingContext = viewModel;
    }

    然后是 Xaml
  • 绑定(bind)到 ObsevableCollection“ItemsPassed”,并将其设置为其中的绑定(bind)上下文。因此,您需要返回
    页面 BindingContext,因此请注意 TapCommand 的绑定(bind)路径。
  • 将 ItemId 作为参数传递


  •     <ListView ItemsSource="{Binding ItemsPassed}"
    HasUnevenRows="True"
    x:Name="ItemListView">
    <ListView.ItemTemplate>
    <DataTemplate>
    <ViewCell>
    <ContentView>
    <StackLayout Padding="20">
    <StackLayout Orientation="Horizontal">
    <StackLayout Orientation="Vertical" HorizontalOptions="Start">
    <Label Text="{Binding ItemText}"
    FontAttributes="Bold"/>
    </StackLayout>
    <Image Source="{Binding Favourited, HeightRequest="12" HorizontalOptions="EndAndExpand">
    <Image.GestureRecognizers>
    <TapGestureRecognizer
    Command="{Binding Path=BindingContext.TapCommand, Source={x:Reference ItemListView}}"
    CommandParameter="{Binding ItemId}"/>
    </Image.GestureRecognizers>
    </Image>
    </StackLayout>
    </StackLayout>
    </ContentView>
    </ViewCell>
    </DataTemplate>
    </ListView.ItemTemplate>
    </ListView>


    查看型号
  • 定义 ICommand 并为其分配函数
  • 在可观察集合中找到该项目并更改属性
  • 这也需要使用 await App.Database.SaveItemAsync(item) 在数据库中进行更改 - 因此,命令调用的函数必须是异步的


  • public class ItemsViewModel : INotifyPropertyChanged
    {
    public ObservableCollection<Item> ItemsPassed { get; set; } = new ObservableCollection<Item>();
    // Get items out of yourdatabase
    public async void GetItems()
    {
    List<Item> itemsOut = await App.Database.GetItemsAsync();
    foreach (Item i in itemsOut)
    {
    ItemsPassed.Add(i);
    }
    }
    ICommand tapCommand;
    public ItemsViewModel()
    {
    GetItems();
    tapCommand = new Command(ExecuteTap);
    }
    public ICommand TapCommand
    {
    get { return tapCommand; }
    }
    // Find the item selected and change the property
    private async void ExecuteTap(object obj)
    {
    var item = ItemsPassed.FirstOrDefault(i => i.ItemId.ToString() == obj.ToString());
    if (item != null)
    {
    if (item.Favourited == true)
    {
    item.Favourited = false;
    }
    else
    {
    item.Favourited = true;
    }
    await App.Database.SaveItemAsync(item);
    Console.WriteLine("update the database");
    }
    }
    }


    然后,您要确保更改发生在 View 中 - 这是通过在模型中扩展 INotifyPropertyChange 并在属性更改时调用它来完成的。

    型号
  • 当 View 模型中的项目发生更改时,会触发 OnPropertyChanged,从而导致 View 更新。
    public class Item : INotifyPropertyChanged

    {
    [主键,自动增量]
    公共(public) int ItemId { 获取;放; }
    公共(public)字符串 ItemText { 获取;放; }
    公共(public)字符串作者 { 获取;放; }
    private bool _favourited;
    public bool Favourited
    {
    get { return _favourited; }
    set
    {
    _favourited = value;
    OnPropertyChanged("Favourited");
    }
    }

    public event PropertyChangedEventHandler PropertyChanged;
    protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
    {
    PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
    }

  • 希望这可以帮助。抱歉,我不得不将代码部分无法正常工作的代码片段放入其中,这可能是因为某些代码是错误的,但这只是举例。

    关于Xamarin:在后台更新 ObservableCollection,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49725953/

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