Это должно быть возможно с помощью функции оболочки SHOpenFolderAndSelectItems
EDIT
Вот пример кода, показывающий, как использовать эту функцию в C/C++, без проверки ошибок:
//Directory to open
ITEMIDLIST *dir = ILCreateFromPath(_T("C:\\"));
//Items in directory to select
ITEMIDLIST *item1 = ILCreateFromPath(_T("C:\\Program Files\\"));
ITEMIDLIST *item2 = ILCreateFromPath(_T("C:\\Windows\\"));
const ITEMIDLIST* selection[] = {item1,item2};
UINT count = sizeof(selection) / sizeof(ITEMIDLIST);
//Perform selection
SHOpenFolderAndSelectItems(dir, count, selection, 0);
//Free resources
ILFree(dir);
ILFree(item1);
ILFree(item2);
В зависимости от то, что Вы на самом деле хотите выполнить Вас, может делать это с Автогорячая клавиша . Это - удивительный бесплатный инструмент для автоматизации вещей, которые Вы обычно не можете делать. Это должно идти с Windows. Этот сценарий выберет Ваш файл и выделит следующие два файла ниже его при ударе F12.
F12::
run explorer.exe /select`, "c:\path\to\file.txt"
SendInput {Shift Down}{Down}{Down}{Shift Up}
return
также возможно просто поместить те две средних строки в текстовый файл и затем передать его, parm к autohotkey.exe. У них есть опция скомпилировать сценарий также, который сделал бы ее автономным exe, который Вы могли назвать. Работает отлично с большим справочным файлом.
@Orion, возможно использовать автогорячую клавишу от C#. Можно превратить сценарий автогорячей клавиши в автономный исполняемый файл (о 400k), который может быть запущен приложением C# (просто способ, которым Вы запускаете проводник). Можно также передать его параметры командной строки. Это не имеет никаких требований во время выполнения.
Это - один из тех вопросов, где может быть хорошо рассмотреть то, чего Вы пытаетесь достигнуть, и существует ли лучший метод.
Для добавления еще некоторого контекста - Наша компания разрабатывает клиентское приложение C#, которое позволяет пользователям загружать файлы и действительно наполняет ими, отчасти как то, как iTunes управляет файлами MP3, не показывая Вам фактический файл на диске.
полезно выбрать файл в приложении и сделать команду 'Show me this file in Windows Explorer' - это - то, чего я пытаюсь достигнуть и сделал так для единственных файлов.
у Нас есть ListView, который позволяет пользователям выбирать несколько файлов в рамках приложения и перемещаться/удалять/и т.д. их. Было бы хорошо иметь, это 'показывает мне этот файл в окнах' работа команды для нескольких выбранных файлов - по крайней мере, если все исходные файлы находятся в том же каталоге, но если это не возможно затем, что это не основная функция.
Я предполагаю, что можно использовать FindWindowEx
, чтобы получить SysListView32 Windows Explorer, затем использовать SendMessage
с LVM_SETITEMSTATE
для выбора объектов. Причем трудность состоит в том, чтобы знать положение объектов... Возможно LVM_FINDITEM
может использоваться для этого.
Grr я хотел бы сделать это также. Media Player делает это, когда Вы выбираете 2 + файлы и щелкаете правой кнопкой и "открываете расположение файла", но не абсолютно уверенные, как (и при этом я действительно не испытываю желание проводить время w/procmon для понимания этого).
Настоящий способ выбора нескольких файлов в Проводнике следующий
Неуправляемый код выглядит следующим образом (составлено из сообщений кода Китая с исправлением его ошибок)
static class NativeMethods
{
[DllImport("shell32.dll", ExactSpelling = true)]
public static extern int SHOpenFolderAndSelectItems(
IntPtr pidlFolder,
uint cidl,
[In, MarshalAs(UnmanagedType.LPArray)] IntPtr[] apidl,
uint dwFlags);
[DllImport("shell32.dll", CharSet = CharSet.Auto)]
public static extern IntPtr ILCreateFromPath([MarshalAs(UnmanagedType.LPTStr)] string pszPath);
[ComImport]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
[Guid("000214F9-0000-0000-C000-000000000046")]
public interface IShellLinkW
{
[PreserveSig]
int GetPath(StringBuilder pszFile, int cch, [In, Out] ref WIN32_FIND_DATAW pfd, uint fFlags);
[PreserveSig]
int GetIDList([Out] out IntPtr ppidl);
[PreserveSig]
int SetIDList([In] ref IntPtr pidl);
[PreserveSig]
int GetDescription(StringBuilder pszName, int cch);
[PreserveSig]
int SetDescription([MarshalAs(UnmanagedType.LPWStr)] string pszName);
[PreserveSig]
int GetWorkingDirectory(StringBuilder pszDir, int cch);
[PreserveSig]
int SetWorkingDirectory([MarshalAs(UnmanagedType.LPWStr)] string pszDir);
[PreserveSig]
int GetArguments(StringBuilder pszArgs, int cch);
[PreserveSig]
int SetArguments([MarshalAs(UnmanagedType.LPWStr)] string pszArgs);
[PreserveSig]
int GetHotkey([Out] out ushort pwHotkey);
[PreserveSig]
int SetHotkey(ushort wHotkey);
[PreserveSig]
int GetShowCmd([Out] out int piShowCmd);
[PreserveSig]
int SetShowCmd(int iShowCmd);
[PreserveSig]
int GetIconLocation(StringBuilder pszIconPath, int cch, [Out] out int piIcon);
[PreserveSig]
int SetIconLocation([MarshalAs(UnmanagedType.LPWStr)] string pszIconPath, int iIcon);
[PreserveSig]
int SetRelativePath([MarshalAs(UnmanagedType.LPWStr)] string pszPathRel, uint dwReserved);
[PreserveSig]
int Resolve(IntPtr hwnd, uint fFlags);
[PreserveSig]
int SetPath([MarshalAs(UnmanagedType.LPWStr)] string pszFile);
}
[Serializable, StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode), BestFitMapping(false)]
public struct WIN32_FIND_DATAW
{
public uint dwFileAttributes;
public FILETIME ftCreationTime;
public FILETIME ftLastAccessTime;
public FILETIME ftLastWriteTime;
public uint nFileSizeHigh;
public uint nFileSizeLow;
public uint dwReserved0;
public uint dwReserved1;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 260)]
public string cFileName;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 14)]
public string cAlternateFileName;
}
public static void OpenFolderAndSelectFiles(string folder, params string[] filesToSelect)
{
IntPtr dir = ILCreateFromPath(folder);
var filesToSelectIntPtrs = new IntPtr[filesToSelect.Length];
for (int i = 0; i < filesToSelect.Length; i++)
{
filesToSelectIntPtrs[i] = ILCreateFromPath(filesToSelect[i]);
}
SHOpenFolderAndSelectItems(dir, (uint) filesToSelect.Length, filesToSelectIntPtrs, 0);
ReleaseComObject(dir);
ReleaseComObject(filesToSelectIntPtrs);
}
private static void ReleaseComObject(params object[] comObjs)
{
foreach (object obj in comObjs)
{
if (obj != null && Marshal.IsComObject(obj))
Marshal.ReleaseComObject(obj);
}
}
}