xamarin.forms - 使用 ReactiveUI 的可分页实时搜索
问题描述
当我需要我的视图可分页时,我有一个添加到我的 ViewModel 的界面
public interface IPageable
{
Task LoadNextPage();
bool IsLoadingPage { get; set; }
}
然后我将 a 附加PageableBehavior
到我的ListView
,每当我的 ListView 到达页面底部时,它就会调用IPageable.LoadNextPage();
ViewModel。(注意:不使用 ReactiveUI 时,我已经完成了所有这些工作)
我的问题是,虽然我可以轻松地对搜索词做出反应,但我似乎无法弄清楚如何以反应方式将下一页加载到_searchResults
.
public class SeedSelectionViewModel : ReactiveObject, IPageable
{
private const int PageSize = 30;
private const int Threshold = 3;
private static readonly TimeSpan _throttle = TimeSpan.FromMilliseconds(200);
private readonly ISeedInputDataService _dataService;
private readonly ObservableAsPropertyHelper<IEnumerable<SeedVarietySeed>> _searchResults;
public SeedSelectionViewModel(ISeedInputDataService dataService)
{
_dataService = dataService;
this.WhenAnyValue(x => x.SearchText)
.Throttle(_throttle)
.Select(term => term?.Trim())
.DistinctUntilChanged()
.Where(term => !string.IsNullOrWhiteSpace(term) && term.Trim().Length >= Threshold)
.SelectMany((searchText,count,token) => SearchSeedVarieties(searchText,0))
.ObserveOn(RxApp.MainThreadScheduler)
.ToProperty(this, nameof(SearchResults), out _searchResults);
}
[Reactive] public string SearchText { get; set; }
[Reactive] public bool IsLoadingPage { get; set; }
public IEnumerable<SeedVarietySeed> SearchResults => _searchResults.Value;
private Task<IEnumerable<SeedVarietySeed>> SearchSeedVarieties(string searchText, int skip)
=> _dataService.FindSeedVarieties(searchText, skip, PageSize, CancellationToken.None);
public Task LoadNextPage()
{
// how do I load additional results into _searchResults?
return Task.CompletedTask;
}
}
解决方案
这里的麻烦是你不能ObservableAsPropertyHelper
以这种方式使用。您不能从两个不同的订阅更改它。
处理它的正确方法是使用一个ObservableCollection<T>
或类似的,并InvokeCommand
从WhenAnyValue
ctor()
{
_loadDataCommand = ReactiveCommand.CreateFromTask<string>(LoadData);
this.WhenAnyValue(x => x.SearchText)
.Throttle(_throttle)
.Select(term => term?.Trim())
.DistinctUntilChanged()
.Where(term => !string.IsNullOrWhiteSpace(term) && term.Trim().Length >= Threshold)
.ObserveOn(RxApp.MainThreadScheduler)
.InvokeCommand(_loadDataCommand);
}
private async Task LoadData(string searchText)
{
var result = await _dataService.FindSeedVarieties(searchText, 0, PageSize, CancellationToken.None).ConfigureAwait(true);
SearchResults.ReplaceRange(result);
}
推荐阅读
- python - Python:从一行中的字符串中删除初始数字和下划线(但保留其他下划线)
- c# - 使用上下文注入替换非绑定类中的 ScenarioContext.Current
- python - 如何显示存储在列表中的图像
- javascript - 运行 ng serve 与 ng build 时 index.html 中的差异加载脚本引用不同
- c# - 从其他脚本停止协程
- javascript - 在 JSON 中获取对象数组并能够访问其每个属性的问题
- visual-studio-code - vscode每次默认目录为随机文件
- elasticsearch - 如何找到数据节点失败的原因?(弹性搜索 6.5)
- python - 如何在熊猫中使用 python 解码 geohash?
- gmail - 在 GSuite 中创建共享支持电子邮件