目标:
我想要做的是将应用程序外部的UWP 应用程序中的 aGridViewItem
拖到桌面资源管理器中。在放置事件期间,我想从 Internet 下载一个文件并创建一个用于填充. 我希望将其复制到桌面。不幸的是,当延迟用于 UWP 拖放(使用)时,只要您离开应用程序窗口,请求就会被激活,您必须使用要传输的对象来填充。所以,在我看来,我需要使用用 .created创建的延迟类型。GridView
StorageFile
DataPackage
StorageFile
SetDataProvider
DataPackage
StorageFile
CreateStreamedFileFromUriAsyc
我不想每次开始进行拖放操作时都预先下载数据。我只想在实际将数据放到合法复制的地方时才下载数据。
StorageFile
我知道如何使用延迟请求 将预先存在的 UWP 拖放到资源管理器(桌面)。- 我也知道如何创建一个仅在请求数据时才 为您提供数据的
StorageFile
使用。CreateStreamedFileFromUriAsyc
StorageFile
当我尝试将这两个想法结合起来时,Windows 资源管理器给了我错误“不支持接口。”。
如果我使用完全相同的代码,但只是通过在延迟拖动处理程序期间调用 GetBasicPropertiesAsync 之类的东西来获取文件内容,那么它只有在我将拖动放在桌面上直到文件下载时才有效。当拖动图标从“禁止”图标变为“复制”时,我可以看到它完成了。如果我在完成之前松开鼠标按钮,则不会发生复制并且不会引发错误。
显然,我希望拖放下载而不必在延迟处理程序上手动启动它。有任何想法吗?这可能吗?
(是的,我意识到创建正确文件扩展名的代码是错误/不完整的,但这在这里无关紧要......)
//DragStarted Handler in constructor
DragItemsStartedCommand = ReactiveCommand.Create<DragItemsStartingEventArgs>((e) =>
{
_dragItems = e.Items.Cast<ItemViewModel>();
e.Data.Properties.Title = "Transfer file";
e.Data.Properties.Description = "desscription of transfering a file";
e.Data.Properties.FileTypes.Add(StandardDataFormats.StorageItems);
e.Data.SetDataProvider(Windows.ApplicationModel.DataTransfer.StandardDataFormats.StorageItems, OnDeferredStorageFileRequestedHandler);
e.Data.RequestedOperation = DataPackageOperation.Copy;
});
//Deferred request handler
async void OnDeferredStorageFileRequestedHandler(DataProviderRequest request)
{
DataProviderDeferral deferral = request.GetDeferral();
try
{
Task<IEnumerable<StorageFile>> task = null;
await Windows.ApplicationModel.Core.CoreApplication.MainView.CoreWindow.Dispatcher.RunAsync(CoreDispatcherPriority.Normal, async () =>
{
task = DownloadStorageFiles();
});
var result = await task;
request.SetData(result);
}
catch (Exception ex)
{
// Handle the exception
}
finally
{
deferral.Complete();
Debug.WriteLine("deferral complete!!!");
}
}
//Create StorageFile with deferred loading Task
async Task<IEnumerable<StorageFile>> DownloadStorageFiles()
{
List<StorageFile> storageItems = new List<StorageFile>();
foreach (var item in _dragItems)
{
var request = new RestSharp.RestRequest();
var defaultItemType = ItemType.MSWord;
switch (item.MimeTypeTranslated)
{
case ItemType.GoogleDocument:
case ItemType.GoogleSpreadsheet:
case ItemType.GooglePresentation:
case ItemType.GoogleDrawing:
case ItemType.GoogleScript:
request.Resource = $"files/{item.File.id}/export";
request.AddParameter("mimeType", Statics.ItemTypeDictionary.First(x => x.Value == defaultItemType).Key);
break;
default:
request.Resource = $"files/{item.File.id}";
request.AddParameter("alt", "media");
break;
}
string fileName = "";
if (item.File.name.EndsWith($".{Statics.ItemExtensionDictionary[defaultItemType]}"))
fileName = $"{item.File.name}";
else
fileName = $"{item.File.name}.{Statics.ItemExtensionDictionary[defaultItemType]}";
var uri = account.Client.GetAuthorizedUriForDownload(request);
var thumbnail = RandomAccessStreamReference.CreateFromFile(await StorageFile.GetFileFromApplicationUriAsync(new Uri("ms-appx:///StoreLogo.png")));
var storageFileDeferred = await StorageFile.CreateStreamedFileFromUriAsync(fileName, uri , thumbnail);
//var props = await storageFileDeferred.GetBasicPropertiesAsync();
storageItems.Add(file);
}
return storageItems;
}
这个问题的 GitHub 重现: https ://github.com/limefrogyank/DragDeferredFileToDesktop
- 第一页是一个常规的拖到桌面的工作,因为 StorageFile(+ 基础数据)已经在 assets 文件夹中
- 第二页显示使用通过 CreateStreamedFileFromUriAsync 创建的 StorageFile 时生成的错误。
- 第三页使用相同类型的 StorageFile,但使用 hack 强制同步检索数据。桌面冻结一秒钟,直到数据准备好。