2016-12-15 2 views
0

Я ограничиваю действия перетаскивания в элемент управления списком WPF в Powershell, чтобы разрешить удаление текстовых файлов. Я хотел бы использовать свойство System.Windows.DragDropEffects, чтобы предотвратить действие drop при событии DragEnter, так как оно также изменяет курсор мыши, предоставляя пользователю обратную связь для действия отклоненного капли. Я все же могу ограничить действие, предпринятое в отброшенном файле, путем проверки расширения файла в событии Drop. Но я бы предпочел предотвратить совместное действие капли для более плавного взаимодействия с пользователем.Проблема с сценарием Powershell для обработки перетаскивания для списка WPF

В отладке Я проверял правильность установки свойства DragDropEffect, однако обработчик события, похоже, не отражает изменения. Я считаю, что это может быть ограничение, пытающееся использовать класс DragEventArgs для мониторинга события через конвейер Powershell.

Код для списка событий WPF. Событие DragEnter находится ниже. Я заметил, что объект, переданный в конвейере $ _, относится к классу System.Windows.DragEventArgs.

$listbox.Add_DragEnter({ 
if ($_.Data.GetDataPresent([Windows.Forms.DataFormats]::FileDrop)) { 
    foreach ($filename in $_.Data.GetData([Windows.Forms.DataFormats]::FileDrop)) { 
     if(([System.IO.Path]::GetExtension($filename).ToUpper() -eq ".TXT")) { 
      $_.Effects = [System.Windows.DragDropEffects]::All 
      Write-Host 'Dropfile is a .TXT' 
     } 
     else { 
      $_.Effects = [System.Windows.DragDropEffects]::None 
      Write-Host 'Dropfile is NOT a .TXT' 
     } 
    } 
} 
}) 

Настройка свойства DragDropEffect с использованием списка WinForms работает должным образом. Мышь изменяется, и событие drop предотвращается. Однако здесь объект, переданный в конвейере $ _, имеет класс System.Windows.Forms.DragEventArgs.

Add-Type -AssemblyName System.Windows.Forms 
Add-Type -AssemblyName System.Drawing 
$form = New-Object Windows.Forms.Form 
$listbox = New-Object Windows.Forms.ListBox 
$listbox.AllowDrop = $true 
$listbox.Add_DragEnter({ 
$_.Effect = [Windows.Forms.DragDropEffects]::None 
}) 

$form.Controls.Add($listbox) 
$form.ShowDialog() 

Полный тест ниже код для WPF:

[System.Reflection.Assembly]::LoadWithPartialName("System.windows.forms") | Out-Null 

[xml]$xaml = @' 
<Window 
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" 
    Title="Remote Execution Toolkit" Height="300" Width="300"> 
<Grid> 
    <ListBox x:Name="listBox" AllowDrop="True" Height="250" HorizontalAlignment="Center" Width="250"> 
     <ListBox.ItemTemplate> 
      <DataTemplate> 
       <CheckBox IsChecked="{Binding RelativeSource={RelativeSource AncestorType={x:Type ListBoxItem}}, Path=IsSelected}"> 
        <TextBlock Text="{Binding}" TextAlignment="Left" Width="Auto" /> 
       </CheckBox> 
      </DataTemplate> 
     </ListBox.ItemTemplate> 
    </ListBox> 
</Grid> 
</Window> 
'@ 

# Load XAML Reader 
$reader=(New-Object System.Xml.XmlNodeReader $xaml) 
$Window=[Windows.Markup.XamlReader]::Load($reader) 

# Map XAML Controls 
$xaml.SelectNodes("//*[@*[contains(translate(name(.),'n','N'),'Name')]]") | ForEach { 
New-Variable -Name $_.Name -Value $Window.FindName($_.Name) -Force 
} 

# Drag Event to validate file extensions for drop effect 
$listbox.Add_DragEnter({ 
if ($_.Data.GetDataPresent([Windows.Forms.DataFormats]::FileDrop)) { 
    foreach ($filename in $_.Data.GetData([Windows.Forms.DataFormats]::FileDrop)) { 
     if(([System.IO.Path]::GetExtension($filename).ToUpper() -eq ".TXT")) { 
      $_.Effects = [System.Windows.DragDropEffects]::All 
      Write-Host 'Dropfile is a .TXT' 
     } 
     else { 
      $_.Effects = [System.Windows.DragDropEffects]::None 
      Write-Host 'Dropfile is NOT a .TXT' 
     } 
    } 
} 
}) 

$Window.ShowDialog() 

Любые мысли или предложения оценили !!

ответ

0

После тонны Google-Foo, запуская Snoop WPF для мониторинга событий, а также проб и ошибок, я понял, что я просто подписался на неправильное событие Drag. Для достижения результата непрерывного отображения операции не допускается курсор, вы должны использовать событие DragOver.

$listbox.Add_DragOver({ 
    ... 
    $_.Effects = [System.Windows.DragDropEffects]::None 
    ... 
}) 

По-видимому, при использовании WPF кода в Powershell, событие DragEnter только срабатывает один раз позволяя курсор, чтобы изменить назад, в то время как событие DragOver непрерывно срабатывает в то время как мыши находится на элементе управления, поддерживая отображение операции не допускается курсора ,

Надеюсь, что это может спасти другого разработчика-разработчика по дороге, сэкономив некоторое время.