Реализация COM-интерфейса IContextMenu с использованием JNA

#java #winapi #com #jna #vtable

#java #winapi #com #jna #vtable

Вопрос:

Мне нужны все элементы из меню оболочки проводника Windows. Я реализую с помощью jna интерфейс COM-объекта IShellFolder. Но теперь у меня проблема с реализацией интерфейса IContextMenu для запроса контекстного меню. Когда я вызываю функцию QueryContextMenu, результат HResult равен 0, как true, но HMenu было пустым, когда я вызывал функцию GetMenuItemCount, и результат равен 0.

Итак, можете ли вы сказать мне, где у меня ошибка. Большое вам спасибо.

Это мой код для IContextMenu:

 public interface IContextMenu {
Guid.IID IID_IContextMenu = new Guid.IID("{000214E4-0000-0000-C000-000000000046}");

WinNT.HRESULT QueryContextMenu(WinDef.HMENU hMenu, int indexMenu, int idCmdFirst, int idCmdLast, int uFlags);
WinNT.HRESULT InvokeCommand(Pointer pici);
WinNT.HRESULT GetCommandString(IntByReference idCmd, int uType, IntByReference pReserved, WTypes.LPSTR pszName, int cchMax);

public static class Converter {
    public Converter() {
    }

    public static IContextMenu PointerToIContextMenu(PointerByReference ptr) {
        final Pointer interfacePointer = ptr.getValue();
        final Pointer vTablePointer = interfacePointer.getPointer(0);
        final Pointer[] vTable = new Pointer[3];
        vTablePointer.read(0, vTable, 0, 3);

        return new IContextMenu() {
            @Override
            public WinNT.HRESULT QueryContextMenu(WinDef.HMENU hMenu, int indexMenu, int idCmdFirst, int idCmdLast, int uFlags) {
                Function f = Function.getFunction(vTable[2], Function.ALT_CONVENTION);
                return new WinNT.HRESULT(f.invokeInt(new Object[] { interfacePointer, hMenu.getPointer(), indexMenu, idCmdFirst, idCmdLast, uFlags }));
            }

            @Override
            public WinNT.HRESULT InvokeCommand(Pointer pici) {
                Function f = Function.getFunction(vTable[1], Function.ALT_CONVENTION);
                return new WinNT.HRESULT(f.invokeInt(new Object[]{ interfacePointer, pici }));
            }

            @Override
            public WinNT.HRESULT GetCommandString(IntByReference idCmd, int uType, IntByReference pReserved, WTypes.LPSTR pszName, int cchMax) {
                Function f = Function.getFunction(vTable[0], Function.ALT_CONVENTION);
                return new WinNT.HRESULT(f.invokeInt(new Object[] { interfacePointer, idCmd, uType, pReserved, pszName, cchMax }));
            }
        };
    }
}
  

}

И это мой код для вызова QueryContextMenu:

 IContextMenu contextMenu = null;
PointerByReference contextMenuPtr = new PointerByReference();
Pointer ppidlsPointer = new Memory(Native.POINTER_SIZE * ppidls.length);
ppidlsPointer.setPointer(0, ppidls[0].getValue());
hResult = psfParentFolder.GetUIObjectOf(null, 1, ppidls[0].getPointer(), new Guid.REFIID(IContextMenu.IID_IContextMenu), new IntByReference(0), contextMenuPtr);
if (!COMUtils.SUCCEEDED(hResult))
   return false;

/* End Section */

/* Begin Get Context Menu Section */

contextMenu = IContextMenu.Converter.PointerToIContextMenu(contextMenuPtr);
WinDef.HMENU hMenu = User32Ex.INSTANCE.CreatePopupMenu();
hResult = contextMenu.QueryContextMenu(hMenu, 0, 1, 30, 0x00000004);
if (!COMUtils.SUCCEEDED(hResult)) {
   int error = Native.getLastError();
   return false;
}

int count = User32Ex.INSTANCE.GetMenuItemCount(hMenu.getPointer());
if (count > 0) {
}

/* End Section */
  

Ответ №1:

Возможно, вы читаете не то, что думаете, из IContextMenuVtbl . Этот код указывает, что вы собираете только 3 указателя на функции:

 final Pointer[] vTable = new Pointer[3];
vTablePointer.read(0, vTable, 0, 3);
  

Однако этот интерфейс (как и большинство COM-интерфейсов) расширяется IUnknown , поэтому автоматически получает методы QueryInterface с индексом 0, AddRef с индексом 1 и Release с индексом 2. Итак, это три метода, которые вы, похоже, вызываете, хотя и с неправильными аргументами.

Из файла заголовка индексы указателей продолжаются:

  • 3: QueryContextMenu
  • 4: InvokeCommand
  • 5: GetCommandString

Вы можете либо прочитать все 6 указателей на функции и сослаться на последние 3, либо попытаться прочитать со смещением в 3 указателя, используя существующую нумерацию (но в обратном порядке!)

Я считаю, что вам также следует добавить extends IUnknown в определение вашего интерфейса.