跨进程实现在Tree中快速定位节点

原创 2005年05月31日 15:53:00

跨进程实现在Tree中快速定位节点

--------------------------------------------------------------------------------
 
      前些日子写软件时,需要实现一个功能,就是在Tree中快速定位节点,比如注册表编辑器左边的Tree,

只要给出Tree中的节点路径(以“/”分隔),就可以快速将树展开,并将当前节点定位到指定的节点。功能的

实现并不难,但稍有些麻烦。原因在于,如果是本进程中的Tree,只要发消息就可以了,但如果是另外一个进

程中的Tree,就要在那个进程中申请内存,将Tree节点的文字复制到这块内存,然后再把这块内存的数据复制

到本进程的一块内存中,才能与指定的节点路径相比较。由于这个功能还有一些可一般化的东西,所以就写了

一个DLL,只要给出Tree的句柄和节点路径,就可以展开这颗树并定位节点。

     DLL源代码如下:
/********************************************************************/
/* 文件名: Tree.cpp                                                 */
/*                                                                  */
/* 功能: 标准 DLL ---- 跨进程展开 SysTreeView32 中指定的节点        */
/*                                                                  */
/* 作者: 卢培培 (goodname008)           时间: 2005.02.18            */
/*                                                                  */
/* BLOG: http://blog.csdn.net/goodname008                           */

/********************************************************************/
#include "stdafx.h"
#include "Tree.h"
#include "commctrl.h"
#include <string>

using namespace std;

BOOL APIENTRY DllMain( HANDLE hModule, DWORD ul_reason_for_call, LPVOID lpReserved)
{
     switch (ul_reason_for_call)
     {
         case DLL_PROCESS_ATTACH:
         case DLL_THREAD_ATTACH:
         case DLL_THREAD_DETACH:
         case DLL_PROCESS_DETACH:
              break;
     }
    return TRUE;
}

/********************************************************************/
/* 功  能: 跨进程展开 SysTreeView32 中指定的节点
/*
/* 参  数: hTreeWnd         SysTreeView32 的句柄
/*          lpszPath        SysTreeView32 中的节点路径(忽略大小写)
/*
/* 返回值: TRUE             成功
/*         FALSE            失败(节点路径不存在时会返回失败, 但仍然展开)
/*
/* 说  明: 在节点路径不存在的情况下, 本函数会尽可能展开存在的节点
/********************************************************************/
TREE_API BOOL APIENTRY ExpandTreeNode(HWND hTreeWnd, LPCSTR lpszPath)
{

     string szPath = lpszPath;

     if (szPath.empty())
         return FALSE;

     DWORD dwProcessID = NULL;
     GetWindowThreadProcessId(hTreeWnd, &dwProcessID);
     if (!dwProcessID)
         return FALSE;

     HANDLE hProcess = NULL;
     hProcess = OpenProcess(PROCESS_VM_OPERATION | PROCESS_VM_READ | PROCESS_VM_WRITE |

PROCESS_QUERY_INFORMATION, FALSE, dwProcessID);

     if (!hProcess)
         return FALSE;
     TVITEM tvItem, *pItem = NULL;
     ZeroMemory(&tvItem, sizeof(TVITEM));
     pItem = (TVITEM *)VirtualAllocEx(hProcess, NULL, sizeof(TVITEM), MEM_COMMIT,

PAGE_READWRITE);

     tvItem.mask = TVIF_TEXT;
     tvItem.cchTextMax = 512;
     tvItem.pszText = (LPSTR)VirtualAllocEx(hProcess, NULL, 512, MEM_COMMIT, PAGE_READWRITE);
     tvItem.hItem = TreeView_GetRoot(hTreeWnd);
     if (!tvItem.hItem)
         return FALSE;

     string szPathNode;
     string::size_type nBackslashPos = -1;
     char szItemText[512] = {'/0'};
     do
     {
         szPathNode = szPath.substr(nBackslashPos + 1, szPath.find('//', nBackslashPos + 1) -

nBackslashPos - 1);
         do
         {
              if (!WriteProcessMemory(hProcess, pItem, &tvItem, sizeof(TVITEM), NULL))
                   return FALSE;

              if (!TreeView_GetItem(hTreeWnd, pItem))
                   return FALSE;

              if (!ReadProcessMemory(hProcess, tvItem.pszText, szItemText, 512, NULL))
                   return FALSE;

              if (lstrcmpi(szPathNode.c_str(), szItemText) == 0)
              {

                   TreeView_SelectItem(hTreeWnd, tvItem.hItem);
                   if (TreeView_Expand(hTreeWnd, tvItem.hItem, TVE_EXPAND))
                   {
                       tvItem.hItem = TreeView_GetChild(hTreeWnd, tvItem.hItem);
                       if (!tvItem.hItem)
                            return FALSE;
                   }
              }
              else
              {
                   tvItem.hItem = TreeView_GetNextSibling(hTreeWnd, tvItem.hItem);
                   if (!tvItem.hItem)
                       return FALSE;
              }

         } while(lstrcmpi(szPathNode.c_str(), szItemText) != 0);

         nBackslashPos = szPath.find('//', nBackslashPos + 1);

     } while(nBackslashPos != -1);

     VirtualFreeEx(hProcess, tvItem.pszText, NULL, MEM_RELEASE);
     VirtualFreeEx(hProcess, pItem, NULL, MEM_RELEASE);
     CloseHandle(hProcess);
     return TRUE;
}

    头文件源代码:

#ifdef TREE_EXPORTS
#define TREE_API __declspec(dllexport)
#else
#define TREE_API __declspec(dllimport)
#endif

TREE_API BOOL APIENTRY ExpandTreeNode(HWND hTreeWnd, LPCSTR lpszPath);

    DEF文件如下:

LIBRARY  Tree

EXPORTS

ExpandTreeNode     @1

调用例程就不再这里给出了,DLL和VC的调用例程都是用.net环境写的。

源代码及调用例程的下载地址:
http://csdngoodname008.51.net/Tree.zip

Flex tree增加,删除,查询并定位节点

直接上代码:

Flex Tree自动定位节点

xmlns:s="library://ns.adobe.com/flex/spark"           xmlns:mx="library://ns.adobe.com/flex/mx" wid...

xPath技术快速定位节点

xPath技术快速定位节点xPath技术快速定位节点 xPath技术概览 为何使用xPath技术 xPath技术解决的问题 xPath本质 在dom4j中使用xPath技术 xPath语法 xPath...
  • jpzhu16
  • jpzhu16
  • 2016年06月12日 22:58
  • 2707

Xml处理——快速定位Xml节点和获取数据

场景一:快速定位到某一个具体的节点,并获取节点数据 public static Node getNode(Document doc, String path) { if (doc != null...

extjs+s2sh实现异步Tree的节点搜索和查找下一个(FindNext)

本文讨论了treefilter实现tree的查询,和后台返回路径的方式,来实现对extjs tree的搜索。   介绍部分内容来源网络,先谢谢该作者。             在任何一个Tre...

使用easyui的Tree 实现无限子节点绑定

因项目需求,要实现tree结构显示数据,没法,只能硬上了。但有懒写CSS样式,同事推荐用easyui. 网络搜索,下载看看先。http://www.jeasyui.com/ 果然和我想的一样,不...

Ext带复选框的Tree实现父子节点联动功能

昨晚有一个朋友问我EXT中带复选框的Tree控件是否可以实现节点被选中时父子节点联动的功能,之前做树形控件,一直都是在用Z-Tree的,速度快,功能强,这样的功能需求对于Z-Tree来说,完全小意思,...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:跨进程实现在Tree中快速定位节点
举报原因:
原因补充:

(最多只允许输入30个字)