Форум программистов, компьютерный форум CyberForum.ru
Наши страницы

C++ Builder

Войти
Регистрация
Восстановить пароль
 
Илья2
1 / 1 / 0
Регистрация: 09.09.2015
Сообщений: 22
#1

Как вызвать системное меню (переименовать, удалить, свойства итд) для нескольких файлов и папок? - C++ Builder

13.11.2015, 21:58. Просмотров 358. Ответов 1
Метки нет (Все метки)

Доброго времени суток!
Нашел на паскале хорошее решение
Pascal
1
GetProperties('C:\1.txt', pt, Self);
Работает как надо. Выполняю команду, вызывается точно такое же меню, как если бы я кликнул правой клавишей мыши по файлу. Я долго искал такое же решение на С++, но не нашел, а паскаль я не понимаю. Мне очень не нравилось, что весь код написан на С++, а одна функция написана на паскале и я переписал все на С++ по собственному стилю.
Использование функции приобрело следующий вид:
C++
1
2
3
4
void __fastcall TForm1::RzButton1Click(TObject *Sender)
{
  Show_System_Popup_Menu_U("C:\\1.txt",  Handle);
}
Я хотел написать собственную версию файлового менеджера. В ListView я поместил файлы и папки, сделал что-то вроде explorer`а (также тестировал готовые решения, мне они не понравились). Сделал фильтр после выполнения которого в ListView появляются отфильтрованные файлы и папки(с директориями, иконками итд). Я могу нажать правой клавишей мышы по Item`у ListView и через Show_System_Popup_Menu_U(...) провести полноценную операцию с выделенным файлом или папкой. Но только с 1 элементом.
А возможно ли провести операцию с несколькими элементами? Как при работе с обычными файлами папками, когда заходим в любую директорию выбираем несколько файлов и папок и удаляем или переносим их в другое место или выполняем зарегистрированную команду в системном меню, например, архивацию?
Возможно ли сделать следующий вызов:
C++
1
2
3
4
5
6
  std::vector<String> Directories;
  Directories.push_back("C:\\2.txt");
  Directories.push_back("C:\\1.txt");
  Directories.push_back("C:\\3\\4.txt");
  Directories.push_back("C:\\3\\5.txt");
  Show_System_Popup_Menu_U(Directories,  Handle);

Исходники:
Для С++ я запускал на Rad Studio XE2( в Project->Options->Directories and Conditions ставил _TCHAR maps to = char), на билдере 6 не запускается.
Паскаль тоже запускал на C++ Rad Studio ХЕ2.
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
#include <vcl.h>
#include <shlobj.h>
 
 
void Show_System_Popup_Menu_U(const char * Directory, void* Handle);
void Show_System_Popup_Menu_U(const char * Directory, int X, int Y, void* Handle);
 
 
LRESULT __stdcall Menu_Callback_U(HWND Wnd, unsigned Msg, WPARAM WParam, LPARAM LParam) {
 
    WPARAM g;
    IContextMenu2 *ContextMenu2;
 
    switch (Msg) {
 
    case WM_CREATE: {
 
            ContextMenu2 = (IContextMenu2*)(PCreateStruct(LParam)->lpCreateParams);
            SetWindowLong(Wnd, GWL_USERDATA, Longint(ContextMenu2));
            return DefWindowProc(Wnd, Msg, WParam, LParam);
        }
    case WM_INITMENUPOPUP: {
            ContextMenu2 = (IContextMenu2*)(GetWindowLong(Wnd, GWL_USERDATA));
            ContextMenu2->HandleMenuMsg(Msg, WParam, LParam);
            return 0;
        }
    case WM_DRAWITEM:
    case WM_MEASUREITEM: {
            ContextMenu2 = (IContextMenu2*)(GetWindowLong(Wnd, GWL_USERDATA));
            ContextMenu2->HandleMenuMsg(Msg, WParam, LParam);
            return 1;
        }
    };
    return DefWindowProc(Wnd, Msg, WParam, LParam);
};
 
// ---------------------------------------------------------------------------
 
HWND Create_Menu_Callback_hwnd_U(const IContextMenu2 * ContextMenu) {
 
    HINSTANCE hInstance = GetModuleHandle(NULL);
 
    const char* IcmCallbackWnd = "System_Popup_Menu_Callback";
 
    TWndClassA WndClass;
    memset(&WndClass, 0, sizeof(WndClass));
    WndClass.lpszClassName = IcmCallbackWnd;
    WndClass.lpfnWndProc = &Menu_Callback_U;
    WndClass.hInstance = hInstance;
    RegisterClass(&WndClass);
    return CreateWindow(IcmCallbackWnd, IcmCallbackWnd, WS_POPUPWINDOW, 0, 0, 0, 0, 0, 0, hInstance, Pointer(ContextMenu));
};
 
// ---------------------------------------------------------------------------
 
void Show_System_Popup_Menu_U(const char * Directory, void* Handle_) {
    TPoint Point;
    GetCursorPos(&Point);
    Show_System_Popup_Menu_U(Directory, Point.X, Point.Y, Handle_);
};
 
// ---------------------------------------------------------------------------
 
void Show_System_Popup_Menu_U(const char * Directory, int X, int Y, void* Handle_) {
 
    HWND Handle = (HWND)Handle_;
 
    PItemIDList PathPIDL;
    Cardinal pchEaten, Attr;
    IShellFolder *Desktop = NULL, *ShellFolder = NULL;
 
    PItemIDList FilePIDL[2];
 
    HRESULT AResult = NULL;
 
    IContextMenu *ICMenu = NULL;
    IContextMenu2 * ICMenu2 = NULL;
 
    IMalloc *M = NULL;
    HWND CallbackWindow = NULL;
    HMENU ShellContextMenu = NULL;
 
    HRESULT CoInit = CoInitializeEx(NULL, COINIT_MULTITHREADED);
 
    try {
        String FileName = ExtractFileName(Directory);
        String CommonDir = ExtractFilePath(Directory);
 
        if (SHGetDesktopFolder(&Desktop) != S_OK)
            RaiseLastOSError();
 
        if (FileName == "") {
 
            if (SHGetSpecialFolderLocation(0, CSIDL_DRIVES, &PathPIDL) != S_OK)
                RaiseLastOSError();
 
            if (Desktop->BindToObject(PathPIDL, NULL, IID_IShellFolder, (void**)&ShellFolder) != S_OK)
                RaiseLastOSError();
 
            ShellFolder->ParseDisplayName(Handle, NULL, StringToOleStr(CommonDir), (unsigned long*)&pchEaten, FilePIDL, (unsigned long*)&Attr);
 
            AResult = ShellFolder->GetUIObjectOf(Handle, 1, (const _ITEMIDLIST * *)FilePIDL, IID_IContextMenu, NULL, (void**)&ICMenu);
 
        }
        else {
 
            if (Desktop->ParseDisplayName(Handle, NULL, //
                StringToOleStr(CommonDir), (unsigned long*)&pchEaten, &PathPIDL, (unsigned long*)&Attr) != S_OK)
                RaiseLastOSError();
 
            if (Desktop->BindToObject(PathPIDL, NULL, IID_IShellFolder, (void**)&ShellFolder) != S_OK)
                RaiseLastOSError();
 
            ShellFolder->ParseDisplayName(Handle, NULL, //
                StringToOleStr(FileName), (unsigned long*)&pchEaten, &FilePIDL[0], (unsigned long*)&Attr);
 
            AResult = ShellFolder->GetUIObjectOf(Handle, 1, //
                (const _ITEMIDLIST * *)FilePIDL, IID_IContextMenu, NULL, (void**)&ICMenu);
 
        };
 
        if (AResult == 0) {
 
            ShellContextMenu = CreatePopupMenu();
 
            int QueryContentRes = ICMenu->QueryContextMenu(ShellContextMenu, 0, 1, 0x7FFF, CMF_EXPLORE);
            bool Is_Context_Menu_Ok = QueryContentRes != -1; // true; // QueryContentRes == 50;
            bool Is_Query_Ok = ICMenu->QueryInterface(&ICMenu2) != -1; // ICMenu->QueryInterface(&ICMenu2) == 0;
 
            if (Is_Context_Menu_Ok && Is_Query_Ok) {
 
                CallbackWindow = Create_Menu_Callback_hwnd_U(ICMenu2);
 
                int PopupMenuResult = 0;
 
                try {
                    PopupMenuResult = TrackPopupMenu(ShellContextMenu, //
                        TPM_LEFTALIGN | TPM_LEFTBUTTON | TPM_RIGHTBUTTON | TPM_RETURNCMD, X, Y, 0, CallbackWindow, NULL);
 
                }
                __finally {
                    ICMenu2 = NULL;
                };
 
                if (PopupMenuResult) {
 
                    int ICmd = PopupMenuResult - 1;
 
                    Winapi::Shlobj::TCMInvokeCommandInfo CMD;
                    memset(&CMD, 0, sizeof(CMD));
 
                    CMD.cbSize = sizeof(CMD);
                    CMD.hwnd = Handle;
                    CMD.lpVerb = MakeIntResourceA(ICmd);
                    CMD.nShow = SW_SHOWNORMAL;
                    if (ICMenu->InvokeCommand(&CMD) != S_OK)
                        RaiseLastOSError();
 
                };
 
            };
        };
 
    }
    __finally {
 
        if (FilePIDL[0]) {
            SHGetMalloc(&M);
            if (M)
                M->Free(FilePIDL[0]);
            M = NULL;
 
        }
 
        if (PathPIDL) {
            SHGetMalloc(&M);
            if (M)
                M->Free(PathPIDL);
            M = NULL;
        };
 
        if (ShellContextMenu)
            DestroyMenu(ShellContextMenu);
 
        if (CallbackWindow)
            DestroyWindow(CallbackWindow);
 
        ICMenu = NULL;
        ShellFolder = NULL;
        Desktop = NULL;
 
        if (CoInit == S_OK)
            CoUninitialize();
 
    };
};
Источник:

Pascal
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
/// /////////////////////////////////////////////////////////////////////////////
//
// ****************************************************************************
// * Unit Name : Unit1
// * Purpose   : Демо отображения системного контекстного меню эксплорера
// * Author    : Александр (Rouse_) Багель
// * Version   : 1.00
// ****************************************************************************
//
 
unit Unit1;
 
interface
 
uses
  Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls,
  // Чтоб все заработало - подключаем вот эти 2 юнита
  ShlObj,
  ActiveX;
 
// interface
 
procedure GetProperties(Path: String; MousePoint: TPoint; WC: TWinControl);
 
// type
// TForm1 = class(TForm)
// Button1: TButton;
// procedure Button1Click(Sender: TObject);
// end;
//
// var
// Form1: TForm1;
//
implementation
 
//
// {$R *.dfm}
 
// Это для работы самого меню, как оконного элемента
function MenuCallback(Wnd: HWND; Msg: UINT; WParam: WParam; LParam: LParam)
  : LRESULT; stdcall;
var
  ContextMenu2: IContextMenu2;
begin
  case Msg of
    WM_CREATE:
      begin
        ContextMenu2 := IContextMenu2(PCreateStruct(LParam).lpCreateParams);
        SetWindowLong(Wnd, GWL_USERDATA, Longint(ContextMenu2));
        Result := DefWindowProc(Wnd, Msg, WParam, LParam);
      end;
    WM_INITMENUPOPUP:
      begin
        ContextMenu2 := IContextMenu2(GetWindowLong(Wnd, GWL_USERDATA));
        ContextMenu2.HandleMenuMsg(Msg, WParam, LParam);
        Result := 0;
      end;
    WM_DRAWITEM, WM_MEASUREITEM:
      begin
        ContextMenu2 := IContextMenu2(GetWindowLong(Wnd, GWL_USERDATA));
        ContextMenu2.HandleMenuMsg(Msg, WParam, LParam);
        Result := 1;
      end;
  else
    Result := DefWindowProc(Wnd, Msg, WParam, LParam);
  end;
end;
 
// Это для создания самого меню, как оконного элемента
function CreateMenuCallbackWnd(const ContextMenu: IContextMenu2): HWND;
const
  IcmCallbackWnd = 'ICMCALLBACKWND';
var
  WndClass: TWndClass;
begin
  FillChar(WndClass, SizeOf(WndClass), #0);
  WndClass.lpszClassName := PChar(IcmCallbackWnd);
  WndClass.lpfnWndProc := @MenuCallback;
  WndClass.hInstance := hInstance;
  Windows.RegisterClass(WndClass);
  Result := CreateWindow(IcmCallbackWnd, IcmCallbackWnd, WS_POPUPWINDOW, 0, 0,
    0, 0, 0, 0, hInstance, Pointer(ContextMenu));
end;
 
 
// implementation
 
procedure GetProperties(Path: String; MousePoint: TPoint; WC: TWinControl);
var
  CoInit, AResult: HRESULT;
  CommonDir, FileName: String;
  Desktop, ShellFolder: IShellFolder;
  pchEaten, Attr: Cardinal;
  PathPIDL: PItemIDList;
  FilePIDL: array [0 .. 1] of PItemIDList;
  ShellContextMenu: HMenu;
  ICMenu: IContextMenu;
  ICMenu2: IContextMenu2;
  PopupMenuResult: BOOL;
  CMD: TCMInvokeCommandInfo;
  M: IMAlloc;
  ICmd: Integer;
  CallbackWindow: HWND;
 
begin
  // Первичная инициализация
  ShellContextMenu := 0;
  Attr := 0;
  PathPIDL := nil;
  CallbackWindow := 0;
  CoInit := CoInitializeEx(nil, COINIT_MULTITHREADED);
  try
    // Получаем пути и имя фала
    CommonDir := ExtractFilePath(Path);
    FileName := ExtractFileName(Path);
    // Получаем указатель на интерфейс рабочего стола
    if SHGetDesktopFolder(Desktop) <> S_OK then
      RaiseLastOSError;
    // Если работаем с папкой
    if FileName = '' then
    begin
      // Получаем указатель на папку "Мой компьютер"
      if (SHGetSpecialFolderLocation(0, CSIDL_DRIVES, PathPIDL) <> S_OK) or
        (Desktop.BindToObject(PathPIDL, nil, IID_IShellFolder,
        Pointer(ShellFolder)) <> S_OK) then
        RaiseLastOSError;
      // Получаем указатель на директорию
      ShellFolder.ParseDisplayName(WC.Handle, nil, StringToOleStr(CommonDir),
        pchEaten, FilePIDL[0], Attr);
      // Получаем указатель на контектсное меню папки
      AResult := ShellFolder.GetUIObjectOf(WC.Handle, 1, FilePIDL[0],
        IID_IContextMenu, nil, Pointer(ICMenu));
    end
    else
    begin
      // Получаем указатель на папку "Мой компьютер"
      if (Desktop.ParseDisplayName(WC.Handle, nil, StringToOleStr(CommonDir),
        pchEaten, PathPIDL, Attr) <> S_OK) or
        (Desktop.BindToObject(PathPIDL, nil, IID_IShellFolder,
        Pointer(ShellFolder)) <> S_OK) then
        RaiseLastOSError;
      // Получаем указатель на файл
      ShellFolder.ParseDisplayName(WC.Handle, nil, StringToOleStr(FileName),
        pchEaten, FilePIDL[0], Attr);
      // Получаем указатель на контектсное меню файла
      AResult := ShellFolder.GetUIObjectOf(WC.Handle, 1, FilePIDL[0],
        IID_IContextMenu, nil, Pointer(ICMenu));
    end;
 
    // Если указатель на конт. меню есть, делаем так:
    if Succeeded(AResult) then
    begin
      ICMenu2 := nil;
      // Создаем меню
      ShellContextMenu := CreatePopupMenu;
      // Производим его наполнение
 
      if Succeeded(ICMenu.QueryContextMenu(ShellContextMenu, 0, 1, $7FFF,
        CMF_EXPLORE)) and Succeeded(ICMenu.QueryInterface(IContextMenu2,
        ICMenu2)) then
        CallbackWindow := CreateMenuCallbackWnd(ICMenu2);
      try
        // Показываем меню
        PopupMenuResult := TrackPopupMenu(ShellContextMenu, TPM_LEFTALIGN or
          TPM_LEFTBUTTON or TPM_RIGHTBUTTON or TPM_RETURNCMD, MousePoint.X,
          MousePoint.Y, 0, CallbackWindow, nil);
      finally
        ICMenu2 := nil;
      end;
      // Если был выбран какой либо пункт меню:
      if PopupMenuResult then
      begin
        // Индекс этого пункта будет лежать в ICmd
        ICmd := Longint(PopupMenuResult) - 1;
        // Заполняем структуру TCMInvokeCommandInfo
        FillChar(CMD, SizeOf(CMD), #0);
        with CMD do
        begin
          cbSize := SizeOf(CMD);
          HWND := WC.Handle;
          lpVerb := MakeIntResourceA(ICmd);
          nShow := SW_SHOWNORMAL;
        end;
        // Выполняем InvokeCommand с заполненной структурой
        AResult := ICMenu.InvokeCommand(CMD);
        if AResult <> S_OK then
          RaiseLastOSError;
      end;
    end;
  finally
    // Освобождаем занятые ресурсы чтобы небыло утечки памяти
    if FilePIDL[0] <> nil then
    begin
      // Для освобождения использем IMalloc
      SHGetMAlloc(M);
      if M <> nil then
        M.Free(FilePIDL[0]);
      M := nil;
    end;
    if PathPIDL <> nil then
    begin
      SHGetMAlloc(M);
      if M <> nil then
        M.Free(PathPIDL);
      M := nil;
    end;
    if ShellContextMenu <> 0 then
      DestroyMenu(ShellContextMenu);
    if CallbackWindow <> 0 then
      DestroyWindow(CallbackWindow);
    ICMenu := nil;
    ShellFolder := nil;
    Desktop := nil;
    if CoInit = S_OK then
      CoUninitialize;
  end;
end;
 
//
// // Пример использования
// procedure TForm1.Button1Click(Sender: TObject);
// var
// pt: TPoint;
// begin
// GetCursorPos(pt);
// GetProperties('E:\Guardant\INSTDRV.INI', pt, Self);
// end;
//
end.
0
Вложения
Тип файла: rar SystemPopupMenuC++.rar (1.8 Кб, 3 просмотров)
Тип файла: rar SystemPopupMenuPascal.rar (2.4 Кб, 1 просмотров)
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
13.11.2015, 21:58
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Как вызвать системное меню (переименовать, удалить, свойства итд) для нескольких файлов и папок? (C++ Builder):

Как вызвать системное контекстное меню файла в Listview - Visual Basic .NET
Всем привет. Суть такова - имеется Listview находящийся в виртуальном режиме и возникла необходимость вызвать системное меню для файла, а...

Нужен диалог для выбора нескольких и файлов, и папок - Delphi
Коллеги, добрый день! Есть задача организовать в диалоговом окне возможность выбора нескольких и файлов, и папок. OpenDialog...

Свойства папок/файлов открываются 3-4минуты( - Windows 8, 8.1
Подскажите, пожалуйста, что нужно сделать, чтобы свойства папок/файлов открывались сразу, а не через 3-5 минут? Заранее благодарю за...

Как можно вызвать стандартное окно для выбора папок? - C++
Из С++ нужно вызвать стандартное окно для выбора папок. Что-то типа GetOpenFileName, только для папок. Господа программеры,...

Как сделать общее системное меню для всех активностей? - Программирование Android
в андроиде пока не сильно ориентируюсь. хочется сделть общее меню на все приложение в таком виде, как на примере не работает. ...

Копирование файлов из нескольких папок - CMD/BAT
Товарищи подскажите. Есть одна большая мега папка. В ней есть очень много других папок. Они все имеют одинаковое название, только...

1
Илья2
1 / 1 / 0
Регистрация: 09.09.2015
Сообщений: 22
29.04.2016, 00:55  [ТС] #2
Я переписал код с паскаля на с++, разобрался в коде и у меня все заработало с мульти выделением. Вопрос решен.
0
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
29.04.2016, 00:55
Привет! Вот еще темы с ответами:

Один экзешник из нескольких папок и файлов? - Python
Помогите собрать программу написанную на питоне с нескольких файлов в один .exe Я конвертировал через cx_freeze (создалось 2 папки и...

Как программно вызвать системное окно Windows Настройка даты/времени - C#
Например, окно &quot;Выполнить&quot; средствами API вызыватся так: SendMessage(FindWindow(&quot;Shell_TrayWnd&quot;, &quot;&quot;), 0x111, 401, 0); а экранная...

Выбор нескольких файлов из разных папок и запись имен в ListBox - C#
Здравствуйте, возникла такая проблема, добавлюю файлы с разных директорий в ListBox, которые в последствии будут записаны в архив, но есть...

Копирование нескольких определённых файлов и папок в одну определённую папку - CMD/BAT
Уважаемые форумчане мне нужен скрипт копирования папок и определенных файлов. Мне нужно скопировать 3 папки расположенных в разным местах...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
2
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.
Рейтинг@Mail.ru