使用JNA从Win32 ListView检索项目文本

时间:2014-04-12 22:34:35

标签: java listview jna

我正在尝试从Win32 ListView控件(SysListView32)检索项目信息(文本就足够了)。我正在使用JNA的sendMessage()来发送LVM_GETITEMTEXT。 SendMessage()接受一个LVITEM结构的指针,如下所示(http://msdn.microsoft.com/en-us/library/windows/desktop/bb774760(v=vs.85).aspx):

typedef struct {
  UINT   mask;
  int    iItem;
  int    iSubItem;
  UINT   state;
  UINT   stateMask;
  LPTSTR pszText;
  int    cchTextMax;
  int    iImage;
  LPARAM lParam;
#if (_WIN32_IE >= 0x0300)
  int    iIndent;
#endif 
#if (_WIN32_WINNT >= 0x0501)
  int    iGroupId;
  UINT   cColumns;
  UINT   puColumns;
#endif 
#if (_WIN32_WINNT >= 0x0600)
  int    piColFmt;
  int    iGroup;
#endif 
} LVITEM, *LPLVITEM;

以下是我对LVITEM结构的Java实现:

    public static class LVITEM extends Structure {
        public WinDef.UINT mask;
        public int iItem; 
        public int iSubItem; 
        public WinDef.UINT state; 
        public WinDef.UINT stateMask; 
        public Pointer pszText;
        public int cchTextMax; 
        public int iImage; 
        public WinDef.LPARAM lParam; 
        public int iIndent; 
        public int iGoupId; 
        public WinDef.UINT cColumns; 
        public WinDef.UINT puColumns; 

        @Override
        protected List getFieldOrder() {
            return Arrays.asList(new String[] { 
"mask",  "iItem",  "iSubItem",  "state", "stateMask", "pszText", "cchTextMax",  "iImage", "lParam",  "iIndent", "iGoupId",  "cColumns", "puColumns" });
        }

        //Constructor
        public LVITEM() { 
            Memory m = new Memory(260); 
            mask = new WinDef.UINT((long)1); //code for LVIF_TEXT
            iItem = 0; 
            iSubItem = 0; //no subitem
            pszText = m.getPointer(0); 
            cchTextMax = 260; 
            iImage = 0; 
            lParam = new WinDef.LPARAM(0); 
            iIndent = 0; 
        }
    }

Java结构的大小是52个字节,应该与C ++结构的大小相匹配。

这是我的User32

public interface User32 extends StdCallLibrary {
            User32 INSTANCE = (User32) Native.loadLibrary("user32", User32.class, W32APIOptions.DEFAULT_OPTIONS);
...
int SendMessage (WinDef.HWND hWnd, int msg, WinDef.WPARAM wparam, LVITEM lvItem);
//Several alternative definitions
//int SendMessage (WinDef.HWND hWnd, int msg, int wparam, Pointer lvItem);
...
}

我声明我的结构:

LVITEM lvi = new LVITEM();
lvi.iItem = itemIdx;  // the zero-based index of the ListView item

根据另外两篇文章的一些建议(见下文),我尝试了不同的SendMessage()实现,结果不同。例如:

int ret = user32.SendMessage(hWnd, User32.LVM_GETITEMTEXT, 0,lvi);

程序使用ListView崩溃第三方软件,但不检索项目的文本。这是结构的转储(设置jna.dump_memory = true):

ListView$LVITEM(auto-allocated@0x3173e08 (52 bytes)) {
  WinDef$UINT mask@0=1
  int iItem@4=0
  int iSubItem@8=0
  WinDef$UINT state@c=0
  WinDef$UINT stateMask@10=0
  Pointer pszText@14=native@0x380338
  int cchTextMax@18=104
  int iImage@1c=0
  WinDef$LPARAM lParam@20=0
  int iIndent@24=0
  int iGoupId@28=0
  WinDef$UINT cColumns@2c=0
  WinDef$UINT puColumns@30=0
}
memory dump
[01000000]
[00000000]
[00000000]
[00000000]
[00000000]
[38033800]
[04010000]
[00000000]
[00000000]
[00000000]
[00000000]
[00000000]
[00000000]

ret= 0

这意味着我没有得到任何回报和

lvi.pszText.getString(0) =  8 8

(奇怪的符号链在这里不能很好地复制)。

例如,如果我这样做:

int ret = user32.SendMessage(hWnd, User32.LVM_GETITEMTEXT, new WPARAM(0),new LPARAM(lvi.getPointer().getLong(0)));

我明白了:

ListView$LVITEM(auto-allocated@0x31733e0 (52 bytes)) {
  WinDef$UINT mask@0=1
  int iItem@4=0
  int iSubItem@8=0
  WinDef$UINT state@c=0
  WinDef$UINT stateMask@10=0
  Pointer pszText@14=native@0x380178
  int cchTextMax@18=104
  int iImage@1c=0
  WinDef$LPARAM lParam@20=0
  int iIndent@24=0
  int iGoupId@28=0
  WinDef$UINT cColumns@2c=0
  WinDef$UINT puColumns@30=0
}
memory dump
[00000000]
[00000000]
[00000000]
[00000000]
[00000000]
[00000000]
[00000000]
[00000000]
[00000000]
[00000000]
[00000000]
[00000000]
[00000000]

(所以结构是空的?)和

ret= 0

lvi.pszText.getString(0) = 05  05  � 8

在这种情况下,带有ListView的第三方软件不会崩溃。

我已成功从ListView控件中检索其他项信息,例如count(使用LVM_GETITEMCOUNT)或所选项(使用LVM_GETSELECTEDCOUNT)。

我已经回顾了几个类似的主题,例如: Retrieving item text with JNA and SendMessage()JNA: Pass Pointer to Structure to SendMessage function of User32.dll as the LPARAM 网上有很多其他帖子并实施了很多建议,但问题仍然存在。

有任何建议可以帮助我解决这个问题,我们将不胜感激。

谢谢

1 个答案:

答案 0 :(得分:1)

如果有人感兴趣(草案版本),我就可以使用它:

    PointerByReference lngProcID;
    int lngProcHandle;
    LVITEM lvi;
    int strSize = 255;
    int result = 0;
    IntByReference byteIO = new IntByReference();
    Pointer lngVarPtr1 = null;Pointer lngMemVar1 = null;
    Pointer lngVarPtr2 = null;Pointer lngMemVar2 = null;
    Pointer lviVarPtr = null;Pointer lviVar = null;
    int lngMemLen1; int lngMemLen2;

    lngProcID = new PointerByReference();
    int ThreadId = user32.GetWindowThreadProcessId(hWnd, lngProcID);

    lngProcHandle = Kernel32.OpenProcess(Kernel32.PROCESS_VM_OPERATION | Kernel32.PROCESS_VM_WRITE | Kernel32.PROCESS_VM_READ, false, lngProcID.getValue());

    lvi = new LVITEM();
    lngMemLen1 = strSize;
    lngMemLen2 = lvi.size(); 

    lngMemVar2 = Kernel32.VirtualAllocEx(lngProcHandle, 0, lngMemLen2, Kernel32.MEM_RESERVE|Kernel32.MEM_COMMIT, Kernel32.PAGE_READWRITE);        

    lvi.cchTextMax = strSize;
    lvi.iItem = itemIdx;
    lvi.iSubItem = 0;
    lvi.mask = User32.LVIF_TEXT;
    lvi.pszText = lngMemVar1;       

    //result  = Kernel32.WriteProcessMemory(lngProcHandle, lngMemVar1, lngVarPtr1, lngMemLen1, byteswritten1);
    result = Kernel32.WriteProcessMemory(lngProcHandle, lngMemVar2, lvi, lngMemLen2, byteIO);

    result = user32.SendMessage (hWnd, User32.LVM_GETITEM, 0, lngMemVar2);

   lngVarPtr1 = new Memory(strSize + 1);
   result = Kernel32.ReadProcessMemory(lngProcHandle, lngMemVar1, lngVarPtr1, lngMemLen1, byteIO);

    result = Kernel32.VirtualFreeEx (lngProcHandle, lngMemVar1, 0, Kernel32.MEM_RELEASE);
    result = Kernel32.VirtualFreeEx (lngProcHandle, lngMemVar2, 0, Kernel32.MEM_RELEASE);        
    result = Kernel32.CloseHandle(lngProcHandle);

    return lngVarPtr1.getWideString(0);