3

写真ライブラリに写真を表示したい。写真を取得してデータをバインドします。

StorageFolder picturesFolder = KnownFolders.PicturesLibrary;
IReadOnlyList<StorageFile> myPictures = await picturesFolder.GetFilesAsync();
var mydata = from file in myPictures select new { Subtitle = "subtitle", Title = "title", Image = this.getImage(file.Path) };

this.DefaultViewModel["Items"] = mydata;

これは、セットBitmapImageのgetImage()です。

private async Task<BitmapImage> getImage(string finename)
{
    StorageFolder picturesFolder = KnownFolders.PicturesLibrary;
    StorageFile file = await picturesFolder.GetFileAsync(fileName);
    var stream = await file.OpenReadAsync();

    var bitmap = new BitmapImage();
    bitmap.SetSource(stream);

    return bitmap;
 }

でも写真は表示されません。非同期機能のせいだと思いますが、解決策がわかりません。私たちを手伝ってくれますか ?

4

2 に答える 2

2

に設定されたデータをどのように使用するかはわかりませんDefaultViewModelが、そうです、asyncメソッドが問題のようです。

あなたがする必要があるのは、どういうわけかへのawait各呼び出しgetImage()です。これを行う1つの方法は、でasyncラムダを使用することselectです。ただし、そのためには、メソッド構文を使用する必要があります。

あなたがそれをするとき、あなたはIEnumerable<Task<a>>(あなたの匿名のタイプはどこにありますかa)持っているでしょう、しかしあなたはただIEnumerable<a>。これを取得するには、Task.WhenAll()(を返しますTask<a[]>)を使用してから、awaitその結果を使用します。

var tasks = myPictures.Select(
    async file => new { Subtitle = "subtitle", Title = "title", Image = await getImage(file.Path) });
var data = await Task.WhenAll(tasks);

これにより、すべてgetImage()のが一度に実行されますが、これは最も効率的なソリューションではない可能性があります。それを望まない場合は、別の解決策が必要になります。

于 2013-01-20T01:04:26.900 に答える
1

svickの解決策はうまくいくようですが、彼/彼女が言ったように、それは最も効率的な解決策ではないかもしれません。ファイル数が不明なフォルダーのより良い解決策は、でデータ仮想化を使用することFileInformationFactory.GetVirtualizedFilesVector()です。これは、コンバーターで最適に機能します。

私が使用したもの:

仮想化ファイルリストの取得とListViewへのバインド

private async void GetPicturesFromGalleryFolder()
{
    var queryOptions = new QueryOptions();
    queryOptions.FolderDepth = FolderDepth.Shallow;
    queryOptions.IndexerOption = IndexerOption.UseIndexerWhenAvailable;
    queryOptions.SortOrder.Clear();

    var sortEntry = new SortEntry {PropertyName = "System.DateModified", AscendingOrder = false};
    queryOptions.SortOrder.Add(sortEntry);
    queryOptions.FileTypeFilter.Add(".png");

    var fileQuery = KnownFolders.PicturesLibrary.CreateFileQueryWithOptions(queryOptions);
    var fileInformationFactory =
        new FileInformationFactory(
            fileQuery,
            ThumbnailMode.PicturesView,
            0,
            ThumbnailOptions.None,
            true);

    MyListView.ItemsSource = fileInformationFactory.GetVirtualizedFilesVector();
}

XAML

<ListView.ItemTemplate>
    <DataTemplate>
        <Image
            Source="{Binding Converter={StaticResource converters:IStorageItemInformationToBitmapImageConverter}"/>
    </DataTemplate>
</ListView.ItemTemplate>

IStorageItemInformationToBitmapImageConverter

public class IStorageItemInformationToBitmapImageConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, string language)
    {
        var fileInfo = value as FileInformation;

        if (fileInfo != null)
        {
            var bi = new BitmapImage();

            // The file is being opened asynchronously but we return the BitmapImage immediately.
            SetSourceAsync(bi, fileInfo);
            return bi;

        }

        return null;
    }

    private async void SetSourceAsync(BitmapImage bi, FileInformation fi)
    {
        try
        {
            using (var stream = await fi.OpenReadAsync())
            {
                await bi.SetSourceAsync(stream);
            }
        }
        catch
        {
            // ignore failure
        }
    }

    public object ConvertBack(object value, Type targetType, object parameter, string language)
    {
        return null;
    }
}
于 2013-01-20T05:27:12.717 に答える