PCB CAM自动化基于Incam 打造,在测试时经常遇到调试障碍,每次自行对功能测试时,生成了exe脚本后,再到Incam里面运行,发现问题,再回来修改代码,非常不爽, 参考Genesis调试运行模式的方式,为了工程组写脚本调试时用着爽,那就让大家爽一下,,扩展了一下InCAM 调试功能(即可以在Visual Studio 中可以加断点边调边执行COM指令), 先采用类似Genesis一样查找JOB内存地址,结果碰壁了, 得寻找另外一种方式,发现InCAM每个窗品标题栏都含有JOB与STEP信息,以这个为出发点就很好解决了.


一.获取JOB名的方法(同时兼容,内挂,外挂,区分Genesis,InCAM)

写好一个脚本,可以外挂运行,也可以内挂运行

        /// <summary>
/// 获取JOB名
/// </summary>
/// <param name="JOB_Address"> 如是外挂使用时:填写JOB基址</param>
/// <returns></returns>
public static string getJOB(int JOB_Address = _JOB_Address, int pid = ) // 97b2 0x2FE623C 99b 0x0307BF84
{
if (pid > ) //pid 大于0为外挂启用
{
if (JOB_Address == ) { JOB_Address = _JOB_Address; }
g.JOB = API.ReadMemorystring(JOB_Address, pid);
}
else
{
if (gPID > )
{
if (g.isInCam())
{
string incamTitel = getWindowAllTiltel(); //获取incam标题 调用下方的函数了 不能像Genesis一样通过基址找到JOB地址呢
if (!string.IsNullOrEmpty(incamTitel))
{
incamTitel = incamTitel.Replace(" ", " ");
var arrList = incamTitel.Split(' ');
g.JOB = arrList[];
g.STEP = arrList[];
}
}
else
{
g.JOB = API.ReadMemorystring(JOB_Address, gPID);
}
}
else
{
if (JOB_Address == ) { JOB_Address = _JOB_Address; }
g.JOB = System.Environment.GetEnvironmentVariable("JOB");
}
}
return (g.JOB);
}
/// <summary>
/// 模糊查找标题 incam标题获得JOB STEP名
/// </summary>
/// <param name="TitlelName">InCAM v4</param>
/// <returns></returns>
public static string getWindowAllTiltel(string TitlelName = "InCAM v4")
{
string result = "";
StringBuilder sb = new StringBuilder();
IntPtr desktopPtr = API.GetDesktopWindow();
IntPtr winPtr = API.GetWindow(desktopPtr, GetWindowCmd.GW_CHILD);
while (winPtr != IntPtr.Zero)
{
winPtr = API.GetWindow(winPtr, GetWindowCmd.GW_HWNDNEXT);
API.GetWindowTextW(winPtr, sb, sb.Capacity);
if (sb.ToString().IndexOf(TitlelName) > -)
{
result = sb.ToString();
winPtr = IntPtr.Zero;
}
}
return result;
}

二.用于查找InCAM标题WindowsAPI

        /// <summary>
/// 获取窗口Text
/// </summary>
/// <param name="hWnd"></param>
/// <param name="lpString"></param>
/// <param name="nMaxCount"></param>
/// <returns></returns>
[DllImport("user32.dll")]
public static extern int GetWindowTextW(IntPtr hWnd, [MarshalAs(UnmanagedType.LPWStr)]StringBuilder lpString, int nMaxCount); /// <summary>
/// 该函数返回桌面窗口的句柄。桌面窗口覆盖整个屏幕。桌面窗口是一个要在其上绘制所有的图标和其他窗口的区域。
/// 【说明】获得代表整个屏幕的一个窗口(桌面窗口)句柄.
/// </summary>
/// <returns>返回值:函数返回桌面窗口的句柄。</returns>
[DllImport("user32.dll", EntryPoint = "GetDesktopWindow", CharSet = CharSet.Auto, SetLastError = true)]
public static extern IntPtr GetDesktopWindow();
/// <summary>
/// 该函数返回与指定窗口有特定关系(如Z序或所有者)的窗口句柄。
/// 函数原型:HWND GetWindow(HWND hWnd,UNIT nCmd);
/// </summary>
/// <param name="hWnd">窗口句柄。要获得的窗口句柄是依据nCmd参数值相对于这个窗口的句柄。</param>
/// <param name="uCmd">说明指定窗口与要获得句柄的窗口之间的关系。该参数值参考GetWindowCmd枚举。</param>
/// <returns>返回值:如果函数成功,返回值为窗口句柄;如果与指定窗口有特定关系的窗口不存在,则返回值为NULL。
/// 若想获得更多错误信息,请调用GetLastError函数。
/// 备注:在循环体中调用函数EnumChildWindow比调用GetWindow函数可靠。调用GetWindow函数实现该任务的应用程序可能会陷入死循环或退回一个已被销毁的窗口句柄。
/// 速查:Windows NT:3.1以上版本;Windows:95以上版本;Windows CE:1.0以上版本;头文件:winuser.h;库文件:user32.lib。
/// </returns>
[DllImport("user32.dll", SetLastError = true)]
public static extern IntPtr GetWindow(IntPtr hWnd, GetWindowCmd uCmd);
    /// <summary>
/// 窗口与要获得句柄的窗口之间的关系。
/// </summary>
public enum GetWindowCmd : uint
{
/// <summary>
/// 返回的句柄标识了在Z序最高端的相同类型的窗口。
/// 如果指定窗口是最高端窗口,则该句柄标识了在Z序最高端的最高端窗口;
/// 如果指定窗口是顶层窗口,则该句柄标识了在z序最高端的顶层窗口:
/// 如果指定窗口是子窗口,则句柄标识了在Z序最高端的同属窗口。
/// </summary>
GW_HWNDFIRST = ,
/// <summary>
/// 返回的句柄标识了在z序最低端的相同类型的窗口。
/// 如果指定窗口是最高端窗口,则该柄标识了在z序最低端的最高端窗口:
/// 如果指定窗口是顶层窗口,则该句柄标识了在z序最低端的顶层窗口;
/// 如果指定窗口是子窗口,则句柄标识了在Z序最低端的同属窗口。
/// </summary>
GW_HWNDLAST = ,
/// <summary>
/// 返回的句柄标识了在Z序中指定窗口下的相同类型的窗口。
/// 如果指定窗口是最高端窗口,则该句柄标识了在指定窗口下的最高端窗口:
/// 如果指定窗口是顶层窗口,则该句柄标识了在指定窗口下的顶层窗口;
/// 如果指定窗口是子窗口,则句柄标识了在指定窗口下的同属窗口。
/// </summary>
GW_HWNDNEXT = ,
/// <summary>
/// 返回的句柄标识了在Z序中指定窗口上的相同类型的窗口。
/// 如果指定窗口是最高端窗口,则该句柄标识了在指定窗口上的最高端窗口;
/// 如果指定窗口是顶层窗口,则该句柄标识了在指定窗口上的顶层窗口;
/// 如果指定窗口是子窗口,则句柄标识了在指定窗口上的同属窗口。
/// </summary>
GW_HWNDPREV = ,
/// <summary>
/// 返回的句柄标识了指定窗口的所有者窗口(如果存在)。
/// GW_OWNER与GW_CHILD不是相对的参数,没有父窗口的含义,如果想得到父窗口请使用GetParent()。
/// 例如:例如有时对话框的控件的GW_OWNER,是不存在的。
/// </summary>
GW_OWNER = ,
/// <summary>
/// 如果指定窗口是父窗口,则获得的是在Tab序顶端的子窗口的句柄,否则为NULL。
/// 函数仅检查指定父窗口的子窗口,不检查继承窗口。
/// </summary>
GW_CHILD = ,
/// <summary>
/// (WindowsNT 5.0)返回的句柄标识了属于指定窗口的处于使能状态弹出式窗口(检索使用第一个由GW_HWNDNEXT 查找到的满足前述条件的窗口);
/// 如果无使能窗口,则获得的句柄与指定窗口相同。
/// </summary>
GW_ENABLEDPOPUP =
}

三.InCAM实现外挂调试效果

可边调试,边操作InCAM

PCB InCAM 获取 JOB STEP 实现外挂脚本调试功能实现的更多相关文章

  1. 怎么用C#获取Scenario step在specflow里

    公司最近在用specflow 这种BDD的模式,但PM还是想把case再存进TestManager里面一份儿一遍后期集成TestManager 自动runcase用.所以我们需要获取每个scenari ...

  2. 获取redis主从复制链SHELL脚本

    获取redis主从复制链SHELL脚本 vi redisnode.sh #!/bin/sh master_host=$ master_port=$ auth=$ #判断输入密码是否为空,为空则转化为' ...

  3. 通过kfed自动获取磁盘信息的小脚本

    通过kfed自动获取磁盘信息的小脚本 编译KFED [oracle@rac lib]$cd $ORACLE_HOME/rdbms/lib [oracle@rac lib]$ pwd /u01/app/ ...

  4. Python获取程序运行目录和脚本目录

    Python获取程序运行目录和脚本目录 import os import sys #获取脚本所在目录 print os.path.split( os.path.realpath( sys.argv[0 ...

  5. shell脚本调试

    转自:http://www.ibm.com/developerworks/cn/linux/l-cn-shell-debug/ 一. 前言 shell编程在unix/linux世界中使用得非常广泛,熟 ...

  6. 【转】Linux Shell脚本调试技术

    本文转载自:https://www.ibm.com/developerworks/cn/linux/l-cn-shell-debug/ Shell脚本调试技术 本文全面系统地介绍了shell脚本调试技 ...

  7. shell脚本调试技术_转

    转自:http://itlab.idcquan.com/linux/SHELL/727128.html 参考:https://linux.cn/article-8045-1.html 本文全面系统地介 ...

  8. Shell脚本调试技术

    http://www.ibm.com/developerworks/cn/linux/l-cn-shell-debug/ 一. 前言 shell编程在unix/linux世界中使用得非常广泛,熟练掌握 ...

  9. shell中脚本调试----学习

    1.使用dos2unix命令处理在windows下开发的脚本 将windows下编辑的脚本放置到linux下执行的情况如下: [root@ks ~]# cat -v nginx.sh #!/bin/b ...

随机推荐

  1. java.lang.NoSuchFieldError: DEFAULT_INCOMPATIBLE_IMPROVEMENTS

    解决方案: 启动类上加@EnableAutoConfiguration(exclude = { FreeMarkerAutoConfiguration.class }) 或者在配置文件添加spring ...

  2. layer iframe层弹出图片

    这个js是在一个layer iframe弹出层中,点击按钮弹出图片

  3. UVA - 11214 Guarding the Chessboard(迭代加深搜索)

    题目: 输入一个n*m的棋盘(n,m<10),某些格子有标记,用最少的皇后守卫(即占据或攻击)所有的标记的格子.输出皇后的个数. 思路: 一开始没有想到用迭代加深搜索,直接dfs结果还没写完就发 ...

  4. TestNG忽略测试

    用@Test(enabled = false) 声明需要被忽略执行的测试方法 package com.janson; import org.testng.annotations.Test; publi ...

  5. Python数据分析与展示(1)-数据分析之表示(2)-NumPy数据存取与函数

    NumPy数据存取与函数 数据的CSV文件存取 CSV文件 CSV(Comma-Separated Value,逗号分隔值) CSV是一种常见的文件格式,用来存储批量数据. 将数据写入CSV文件 np ...

  6. 洛谷 2176 [USACO14FEB]路障Roadblock

    [题意概述] 修改图中任一一条边的边权,使其加倍,问怎样使修改后图中的1~n的最短路最大.输出最短路的增量. [题解] 先跑一遍dijkstra求出1~n的路径长度,记录下经过的边.枚举这些边进行修改 ...

  7. 关于markdown(typora)的操作指南,以及导出为word格式文件插件(pandoc-2.6-windows-x86_64)的下载

    Markdown简介 插件链接文章结尾处 目录 Markdown简介1. Markdown是什么?2. 谁创造了它?3. 为什么要使用它?4. 怎么使用?4.1 标题4.2 段落4.3 区块引用4.4 ...

  8. java中List遍历删除元素-----不能直接 list.remove()

    https://blog.csdn.net/github_2011/article/details/54927531 这是List接口中的方法,List集合调用此方法可以得到一个迭代器对象(Itera ...

  9. JVM即时编译(JIT)

    Java解释执行过程: 代码装入-代码校验-代码执行 Java字节码的执行方式分为两种:即使编译方式和解释执行方式.即时编译是值解释器先将字节码编译成机器码,然后执行该机器码.解释执行的方式是指解释器 ...

  10. log4j2.xml 的配置 及使用

     log4j2.xml配置 <?xml version="1.0" encoding="UTF-8"?> <Configuration > ...