Выберите файл для переименования в контекстном меню SharpShell
Я использую SharpShell, чтобы написать крошечный новый пункт контекстного меню оболочки, который копирует выбранные в данный момент файлы в новую подпапку, а затем запрашивает у пользователя новое имя каталога.
В поисках Stackru я нашел этот ответ. Тем не менее, я бы хотел сделать то же самое в SharpShell.
Мне как-то придется уволить SVSI_EDIT
на это, которое я могу найти глубоко в SharpShell.Interop
, но я не уверен, как все это работает. Я не могу найти какую-либо документацию или примеры кода вообще.
(Изменить: я думаю, узнав, как получить Pidl
с имени файла было бы хорошим началом, но, может быть, мне это совсем не нужно?)
2 ответа
Вы можете начать создавать проект с SharpShell, чтобы зарегистрировать новое контекстное меню оболочки, как в этом руководстве.
Здесь мы должны определить класс, реализующий SharpContextMenu
, Для простоты мы создадим меню для любого типа файла и всегда показываем его:
[ComVisible(true)]
[COMServerAssociation(AssociationType.AllFiles)]
public class CopyFilesExtension : SharpContextMenu
{
protected override bool CanShowMenu()
{
return true;
}
protected override ContextMenuStrip CreateMenu()
{
var menu = new ContextMenuStrip();
var copyFiles = new ToolStripMenuItem { Text = "Copy Files To Folder..." };
copyFiles.Click += (sender, args) => CopyFiles();
menu.Items.Add(copyFiles);
return menu;
}
private void CopyFiles()
{
...
}
}
Но я уверен, что вы сделали все это, проблема здесь заключается в реализации CopyFiles()
метод.
Один из способов сделать это - показать диалоговое окно с запросом имени папки, примерно так:
Затем реализовать CopyFiles()
вот так:
private void CopyFiles()
{
using (var dialog = new CopyFileDialog())
{
if (dialog.ShowDialog() == DialogResult.OK)
{
var folder = Path.GetDirectoryName(SelectedItemPaths.First());
var newFolder = Path.Combine(folder, dialog.FolderName);
Directory.CreateDirectory(newFolder);
foreach (var path in SelectedItemPaths)
{
var newPath = Path.Combine(newFolder, Path.GetFileName(path));
File.Move(path, newPath);
}
}
}
}
В приведенном выше коде мы запросили имя папки, затем создали папку и, наконец, переместили выбранные файлы в эту папку.
Однако, если вы хотите сделать это с помощью команды Rename в Windows Explorer, мы можем начать с импорта некоторых необходимых функций Win32:
class Win32
{
[DllImport("shell32.dll", CharSet = CharSet.Unicode)]
public static extern IntPtr ILCreateFromPath([In, MarshalAs(UnmanagedType.LPWStr)] string pszPath);
[DllImport("shell32.dll")]
public static extern int SHOpenFolderAndSelectItems(IntPtr pidlFolder, uint cidl, IntPtr[] apidl, int dwFlags);
[DllImport("shell32.dll")]
public static extern void ILFree(IntPtr pidl);
}
ILCreateFromPath
позволяет нам получить PIDL из имени файла.SHOpenFolderAndSelectItems
Позвольте нам выбрать файл и отправить команду переименования.ILFree
освобождает неуправляемыйPIDL
создано.
С этими функциями Win32 мы можем определить CopyFiles()
следующее:
private void CopyFiles()
{
var folder = Path.GetDirectoryName(SelectedItemPaths.First());
var newFolder = Path.Combine(folder, "New Folder");
Directory.CreateDirectory(newFolder);
foreach (var path in SelectedItemPaths)
{
var newPath = Path.Combine(newFolder, Path.GetFileName(path));
File.Move(path, newPath);
}
RenameInExplorer(newFolder);
}
private static void RenameInExplorer(string itemPath)
{
IntPtr folder = Win32.ILCreateFromPath(Path.GetDirectoryName(itemPath));
IntPtr file = Win32.ILCreateFromPath(itemPath);
try
{
Win32.SHOpenFolderAndSelectItems(folder, 1, new[] { file }, 1);
}
finally
{
Win32.ILFree(folder);
Win32.ILFree(file);
}
}
Мы не можем использовать SharpShell.Interop.Shell32
поскольку единственный метод, доступный в этом классе, ShellExecuteEx()
который используется для запуска новых процессов.
С использованием SelectItemInExplorer
функциональность из приведенного примера в вопросе, очень простая реализация будет выглядеть следующим образом. Там, где это было возможно, любые функции P/Invoke были переписаны так, чтобы использовать как можно больше существующих объявлений SharpShell.
using SharpShell.Attributes;
using SharpShell.Interop;
using SharpShell.SharpContextMenu;
using System;
using System.IO;
using System.Linq;
using System.Runtime.InteropServices;
using System.Windows.Forms;
namespace SendToFolderRename
{
[ComVisible(true)]
[COMServerAssociation(AssociationType.AllFiles)]
public class SendToFolderRename : SharpContextMenu
{
[DllImport("ole32.dll")]
private static extern int CreateBindCtx(int reserved, out IntPtr ppbc);
[DllImport("shell32.dll")]
private static extern int SHOpenFolderAndSelectItems(IntPtr pidlFolder, uint cidl, IntPtr[] apidl, int dwFlags);
protected override bool CanShowMenu()
{
return true;
}
protected override ContextMenuStrip CreateMenu()
{
var menu = new ContextMenuStrip();
var itemCountLines = new ToolStripMenuItem
{
Text = "Copy files to subfolder"
};
itemCountLines.Click += CopyFilesToSubfolder;
menu.Items.Add(itemCountLines);
return menu;
}
private void CopyFilesToSubfolder(object sender, EventArgs e)
{
//System.Diagnostics.Debugger.Break();
string firstSelectedFile = SelectedItemPaths.FirstOrDefault();
if (string.IsNullOrEmpty(firstSelectedFile))
return;
string currentDirPath = (new FileInfo(firstSelectedFile)).DirectoryName;
string newDirName = Path.GetRandomFileName();
string newDirPath = Path.Combine(currentDirPath, newDirName);
DirectoryInfo newDir = Directory.CreateDirectory(newDirPath);
foreach (string filePath in SelectedItemPaths)
{
FileInfo fileInfo = new FileInfo(filePath);
string newFilePath = Path.Combine(fileInfo.DirectoryName, newDirName, fileInfo.Name);
File.Copy(filePath, newFilePath);
}
SelectItemInExplorer(IntPtr.Zero, newDirPath, true);
}
public static void SelectItemInExplorer(IntPtr hwnd, string itemPath, bool edit)
{
if (itemPath == null)
throw new ArgumentNullException("itemPath");
IntPtr folder = PathToAbsolutePIDL(hwnd, Path.GetDirectoryName(itemPath));
IntPtr file = PathToAbsolutePIDL(hwnd, itemPath);
try
{
SHOpenFolderAndSelectItems(folder, 1, new[] { file }, edit ? 1 : 0);
}
finally
{
Shell32.ILFree(folder);
Shell32.ILFree(file);
}
}
private static IntPtr GetShellFolderChildrenRelativePIDL(IntPtr hwnd, IShellFolder parentFolder, string displayName)
{
IntPtr bindCtx;
CreateBindCtx(0, out bindCtx);
uint pchEaten = 0;
SFGAO pdwAttributes = 0;
IntPtr ppidl;
parentFolder.ParseDisplayName(hwnd, bindCtx, displayName, ref pchEaten, out ppidl, ref pdwAttributes);
return ppidl;
}
private static IntPtr PathToAbsolutePIDL(IntPtr hwnd, string path)
{
IShellFolder desktopFolder;
Shell32.SHGetDesktopFolder(out desktopFolder);
return GetShellFolderChildrenRelativePIDL(hwnd, desktopFolder, path);
}
}
}