Я работаю над простым приложением UWP, написанным на C ++ / WinRT под Windows 10, которое содержит два ListView
элемента управления. Цель этого приложения - научиться выбирать элемент из одного ListView
элемента управления, перетаскивать его в другой ListView
элемент управления и отбрасывать элемент так, чтобы он был скопирован из исходного ListView
элемента управления в целевой ListView
элемент управления.
Все примеры, которые я нашел до сих пор, используют C #, а некоторые используют C ++ / CX, а не C ++ / WinRT и собственный C ++, однако мне удалось продвинуться до точки, когда базовая механика выбора элемента из источника ListView
работает как выполняет перетаскивание в место назначения ListView
. Однако при попытке получить информацию из события перетаскивания, чтобы обновить место назначения ListView
, я получаю исключение.
Вопрос: Какие изменения мне нужно внести, чтобы выделенный текст в исходном ListView
элементе управления можно было перетащить на целевой ListView
элемент управления, а затем добавить текст в целевой ListView
элемент управления?
В окне вывода Visual Studio 2017 отображается следующий текст, который я интерпретирую как исключение неверного адреса:
Unhandled exception at 0x0259DC3C (Windows.UI.Xaml.dll) in TouchExperiment_01.exe: 0xC000027B: An application-internal exception has occurred (parameters: 0x05F5E3D8, 0x00000005).
Unhandled exception at 0x74ECE61D (combase.dll) in TouchExperiment_01.exe: 0xC0000602: A fail fast exception occurred. Exception handlers will not be invoked and the process will be terminated immediately.
Unhandled exception at 0x74F9D7D9 (combase.dll) in TouchExperiment_01.exe: Stack cookie instrumentation code detected a stack-based buffer overrun.
Unhandled exception at 0x74F9D7D9 (combase.dll) in TouchExperiment_01.exe: Stack cookie instrumentation code detected a stack-based buffer overrun.
Исключение возникает, когда выполняется следующая строка исходного кода в функции void MainPage::OnListViewDrop()
, которая является последней функцией в исходном файле MainPage.cpp:
auto x = e.DataView().GetTextAsync();
Дополнительная информация A: Используя отладчик, я обнаружил, что сообщение об ошибке связано с исключением, которое подразумевает ошибку в данных, предоставленных методом OnListViewDragItemsStarting()
. Текст сообщения об ошибке исключения:
{m_handle={m_value=0x05550330 L"DataPackage does not contain the specified format. Verify its presence using DataPackageView.Contains or DataPackageView.AvailableFormats." } }
Я также нашел на сайте, где исключение сначала генерируется и перехватывается Visual Studio, останавливая приложение в base.h
(источник из шаблонов C ++ / WinRT), текст ошибки 0x8004006a : Invalid clipboard format
, указывающий на то, что у меня нет согласия по формату данных. что начало перетаскивания создает, а капля перетаскивания пытается поглотить.
Обзор исходного кода
Я изменил стандартный шаблон приложения C ++ / WinRT в области MainPage.xml, MainPage.cpp, MainPage.h и pch.h. Я также добавил файлы классов для нового класса DataSource, который использует std::vector<>
для хранения некоторых тестовых данных. Эти резидентные данные в памяти инициализируются некоторыми фиктивными данными в конструкторе App
:
App::App()
{
InitializeComponent();
DataSource::InitializeDataBase();
Suspending({ this, &App::OnSuspending });
// … other code
Прежде всего мне пришлось добавить строку в файл pch.h, чтобы предоставить шаблоны для перетаскивания:
#include "winrt/Windows.ApplicationModel.DataTransfer.h" // ADD_TO: need to add to allow use of drag and drop in MainPage.cpp
Исходный файл XAML содержит источник для двух элементов управления ListView
, а также элемент управления TextBlock
, который отображает полное описание элемента, выбранного в источнике ListView
:
<Page
x:Class="TouchExperiment_01.MainPage"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:TouchExperiment_01"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d">
<StackPanel Orientation="Horizontal" HorizontalAlignment="Center" VerticalAlignment="Center" Width="1130" Margin="0,0,0,0">
<ListView x:Name="myList" HorizontalAlignment="Left" Height="100" VerticalAlignment="Top" Width="300" SelectionChanged="OnSelectionChanged"
CanDragItems="True" DragItemsStarting="OnListViewDragItemsStarting" BorderBrush="AliceBlue" BorderThickness="3">
</ListView>
<TextBlock x:Name="myTextBlock" Height="200" Width="200" Text="this is temp text to replace." TextWrapping="WrapWholeWords" Margin="5"/>
<ListView x:Name="myList2" HorizontalAlignment="Right" Height="100" VerticalAlignment="Top" Width="300" SelectionChanged="OnSelectionChanged" AllowDrop="True"
DragOver="OnListViewDragOver" Drop="OnListViewDrop" BorderBrush="DarkGreen" BorderThickness="5">
</ListView>
</StackPanel>
</Page>
Объявление класса для DataSource
простое. Определение класса выглядит следующим образом:
#pragma once
class DataSource
{
public:
DataSource();
~DataSource();
static int InitializeDataBase();
struct DataSourceType
{
std::wstring name;
std::wstring description;
};
static std::vector<DataSourceType> myDataBase;
}
;
и инициализация vector
, которая выполняется при создании App
при запуске приложения:
int DataSource::InitializeDataBase()
{
myDataBase.clear();
for (int i = 0; i < 50; i++) {
DataSourceType x;
wchar_t buffer[256] = { 0 };
swprintf_s(buffer, 255, L"Name for %d Item", i);
x.name = buffer;
swprintf_s(buffer, 255, L"Description %d. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam.", i);
x.description = buffer;
myDataBase.push_back(x);
}
return 0;
}
Исходный код MainPage.cpp за страницей XAML:
#include "pch.h"
#include "MainPage.h"
#include "DataSource.h"
using namespace winrt;
using namespace Windows::UI::Xaml;
namespace winrt::TouchExperiment_01::implementation
{
MainPage::MainPage()
{
InitializeComponent();
// load up the source ListView with the name field from out
// in memory database.
auto p = myList().Items();
for (auto a : DataSource::myDataBase) {
p.Append(box_value(a.name));
}
// add a single ListViewItem to the destination ListView so that we
// know where it is.
p = myList2().Items();
p.Append(box_value(L"list"));
}
int32_t MainPage::MyProperty()
{
throw hresult_not_implemented();
}
void MainPage::MyProperty(int32_t /* value */)
{
throw hresult_not_implemented();
}
void MainPage::OnSelectionChanged(Windows::Foundation::IInspectable const & target, Windows::UI::Xaml::RoutedEventArgs const & )
{
// the user has selected a different item in the source ListView so we want to display
// the associated description information for the selected ListViewItem.
winrt::Windows::UI::Xaml::Controls::ListView p = target.try_as<winrt::Windows::UI::Xaml::Controls::ListView>();
if (p) {
int iIndex = p.SelectedIndex();
myTextBlock().Text(DataSource::myDataBase[iIndex].description);
}
}
void MainPage::OnListViewDragItemsStarting(Windows::Foundation::IInspectable const & target, Windows::UI::Xaml::Controls::DragItemsStartingEventArgs const & e)
{
// provide the data that we have in the ListView which the user has selected
// to drag to the other ListView. this is the data that will be copied from
// the source ListView to the destination ListView.
auto p = target.try_as<winrt::Windows::UI::Xaml::Controls::ListView>();
if (p) {
int iIndex = p.SelectedIndex();
e.Items().SetAt(0, box_value(iIndex));
}
}
void MainPage::OnListViewDragOver(Windows::Foundation::IInspectable const & target, Windows::UI::Xaml::DragEventArgs const & e)
{
// indicate that we are Copy of data from one ListView to another rather than one of the other
// operations such as Move. This provides the operation type informative user indicator when the
// user is doing the drag operation.
e.AcceptedOperation(Windows::ApplicationModel::DataTransfer::DataPackageOperation::Copy);
}
void MainPage::OnListViewDrop(Windows::Foundation::IInspectable const & target, Windows::UI::Xaml::DragEventArgs const & e)
{
// update the destination ListView with the data that was dragged from the
// source ListView.
auto p = target.try_as<winrt::Windows::UI::Xaml::Controls::ListView>();
auto x = e.DataView().GetTextAsync(); // ** this line triggers exception on drop.
}
}
Снимок экрана приложения с элементом, выбранным в источнике ListView
до начала перетаскивания, выглядит следующим образом. Исходный ListView
элемент управления находится слева, а целевой ListView
элемент управления - справа.
Приложение: справочные материалы и документация
Microsoft Docs - пространство имен Windows.ApplicationModel.DataTransfer
Microsoft Docs - класс DragItemsStartingEventArgs, который содержит ссылку на этот пример проекта, который, похоже, использует C ++ / CX Перетащите образец на GitHub, который содержит Windows-universal-samples / Samples / XamlDragAndDrop / cpp / Scenario1_ListView.xaml.cpp с полезным примером.