PLM之家PLMHome-国产软件践行者

Catia二次开发CAA中的交互操作开发原理和过程大全

[复制链接]

2018-1-16 16:42:12 6508 0

admin 发表于 2018-1-16 16:42:12 |阅读模式

admin 楼主

2018-1-16 16:42:12

请使用QQ关联注册PLM之家,学习更多关于内容,更多精彩原创视频供你学习!

您需要 登录 才可以下载或查看,没有账号?注册

x
CAA中的交互操作开发,主要包括原理说明,开发过程说明和案例探究。9 q  ^$ q1 Y% z* z$ ]6 m

. t( ]% f8 Y' G* b1 G% _参考阅读:
( y% N1 X6 @. y* L! ]# w% \) w百科全书->UserInterFace->WinTopCommand
) x5 t1 }7 K$ a. n5 Z1 O
/ W* S; v& f& x6 O% M, f. s01.CATIA命令' ^; Y% C% s/ I: `5 @5 z8 O; L

$ A) z* j  f7 E& H4 u5 `, w. tCATIA中命令主要有三种:单步命令,对话框命令,状态转换命令。
- a0 V+ i. x# P! ?  F: L•(1)基本命令,也称(basicCommands):运行时用户不能有附加选项,从其开始运行直至其结束期间无法停止,也称单步命令,该类命令派生自CATCommand类。•(2)对话框命令(DialogBoxCommands):用户可以输入参数值或选择选项,对话框本身即命令,而不是其它命令的一部分,该类命令派生自CATDlgDialog类。•(3)状态对话命令(StateChartCommands):状态对话命令被模拟为状态机,通过状态、迁移(或转换)的组合可构成高级对话命令。命令中可有数个状态,每个状态让用户选择对象、输入参数或选择选项。根据选择的对象、输入参数或选项判断是否满足条件,如果满足相应条件则触发迁移,跳转到下一状态执行,直到命令结束。对话框可用于状态对话命令作参数或选项输入界面。该类命令派生自CATStateCommand类。+ [+ I5 {1 `& U& z. M1 \1 [% d

- O7 c" b+ C; C; f: I) @9 x本文重点说明第三种命令,即状态命令。
  l  G8 E+ e5 O, D  A: H; }( S8 j. L- _, d9 R% g- ]1 S7 r
1 _4 j7 i) G1 Z3 E4 F. ^
在CATIA操作中,交互操作情景很多,如点击屏幕画点、线,
+ o" {8 B) }) ?+ x/ ^/ l: c6 G9 U1 A1 L2 B, d, u
如交互方式输入数值创建特征,
# Z) Q. k7 k$ D  e& R
9 S  I6 y) t* s$ r如用鼠标点选已有模型特征......
( O1 V: }7 k8 x) @# K2 q& Y; t- u0 q: G& _" a
在这些交互操作中,前一步操作往往作为下一步操作的输入或者执行条件,因此需要程序识别每次操作的状态。而维持这一过程有序进行的正是状态机制(satemachine).5 ~4 t0 U# N/ A* v/ L
02.状态机制
/ |5 R; ^# ?3 I0 j7 v/ T8 e
  t) r& n5 k' W" U) R•状态机制(state-machine)是由状态和迁移组成的图,通常状态机附属于类,描述了类实例对接收事件的响应。•状态机是某个类的对象所有可能生命历史的模型,所有外部世界对对象的影响被总结为事件。从命令激活到最后取消,命令生命周期中所有事件都可以用状态机表示。
- ]4 d. W1 n9 \& l, q如上图中点击屏幕获取三点并绘制成折线的状态图可表示如下: [* X/ _6 w' ]) P& L
, h' Q+ m  T7 j. t: }. ]# [" z$ J
•事件是具有时间和空间位置的显著发生的某件事,如鼠标在窗口某个位置点击、控件的某个操作,鼠标移动等。 ; k6 d; v/ t* ~1 u5 I8 X( `
•当状态机检测到事件,将对事件作出判断,并以相关于当前状态的方式来响应,这里的判断称为迁移条件。响应可能包括动作的执行和改变到新的状态。
3 K1 q# [8 j$ q- Y
( }+ C: c! c' R因此,上图中状态机制运行如下:状态机制从开始状态进行到结束状态。从开始状态进入到状态1的转移中无任何函数响应调用,命令自动跳转到状态1的执行,即在鼠标点击事件执行后,在鼠标点击位置创建一个点,只要用户指定一个有效的点,当鼠标左键按下去的时候,鼠标按键事件就已经被检测出,并判断满足创建点的条件,执行响应函数绘制了一个点,同时状态就从第一个状态转移到第二个状态即状态2,第二个状态也是等待用户用鼠标指定一个点。当鼠标第二次按下去的时候,创建第二个点,同时状态就从状态2转移到状态3,鼠标按键事件已经被检测出,直至三个点创建完成,判断了绘制折线的条件,执行了响应函数绘制了一条折线,同时状态从第三个状态转移到最终状态,即结束操作。
( N/ j! @& l/ }4 S* ]) U
7 C4 v* C/ I' W7 q7 F& w- k03.状态命令创建  S5 `* A( w) z1 B( O$ y) a" R

4 p$ `, U3 M# I0 N# V状态命令主要继承于CATStateCommand,该命令类主要用于接收CATIA中用户的交互输入。; w" [! d  V; f8 J0 f. p. \
一个状态命令的执行是基于三个工具而完成的,即代理(Agents),状态(States),转移(Transitions)。
5 W' }8 {9 e6 K  |* n$ Q% \: vAgents:获取用户操作的事件及选择的对象
8 ?! V7 w) \( J$ G3 ^: @States:对话框中等待用于输入
; q0 b; J% j$ F- g) Y9 WTransitions:定义哪些指令将被执行) v3 m. i- m. s" r8 y& ?) q
0 v% V* E* c! [7 v+ |+ E
状态命令执行条件:3 F1 P( v0 A& g( O
1.为每个状态定义一个输入类型
# L6 m5 @% l5 I! }2.一个状态转移需定义由源状态到目标状态的转变
$ v' a6 w1 U' `; ?* @3 n% f" f3.基于条件的状态触发:包括最终事件驱动——用户的交互操作,条件确认用户输入。当某个状态被触发时,目标状态将被激活。
0 `: t& x. L% I( a4.状态转移过程中,可以执行一个或多个指令(Action)4 N2 w% z' i& O- F7 H1 E4 Y& p  T
状态命令开发说明:( ?7 I0 o# h7 @0 e/ U" z
1.创建一个新类(Class)继承于CATStateCommand
* R  ^" m' a+ U8 Y: h9 q2.至少覆盖:描述自己的状态(State)的变迁:BuildGraph()# J7 _6 |( x. D+ M5 \3 B; E
     适时管理命令的生命周期:Activate()、Desactivate()、Cancel()
& S# Z. D) D* ]3 S* D% P( Q4 {% S, g3.定义一些方法,作为条件的验证及指令(Action)的执行+ ~: @0 ~: b2 [/ W5 {
4.保存命令对话框中agent的数据成员% M9 L, |8 W" U  H# E1 _

) T% R$ M  a) d/ _) `+ S$ u& g( ~; G% @" ^; q! M
状态命令执行条件:
+ }5 B* |. e) b" u: j# M! S1.为每个状态定义一个输入类型
7 c4 N% @5 o1 _! G' G3 D) u2.一个状态转移需定义由源状态到目标状态的转变
2 x1 K, w" R% @1 |: d3.基于条件的状态触发:包括最终事件驱动——用户的交互操作,条件确认用户输入。当某个状态被触发时,目标状态将被激活。
4 p! D1 P% A7 t' K% X; P4.状态转移过程中,可以执行一个或多个指令(Action)
  N1 ?3 A2 e: @- K7 ?: n状态命令开发说明:
, ~. A+ A5 d* S( ?2 n1.创建一个新类(Class)继承于CATStateCommand0 d$ s8 M$ n& S7 f) J7 w$ [
2.至少覆盖:描述自己的状态(State)的变迁:BuildGraph()4 `0 l4 P7 s' J1 s, y8 X
     适时管理命令的生命周期:Activate()、Desactivate()、Cancel()
/ ~. t0 {- G  ]; ^9 h3.定义一些方法,作为条件的验证及指令(Action)的执行
- ]7 o9 M. T2 E7 {) x& a+ k) g4.保存命令对话框中agent的数据成员
, Z" H/ y$ z4 F# g9 W: V
; |, k# ?, L+ y. ?4 W& [本文将重点说明三个概念(代理,状态和转移)及其在CAA开发中的体现。(部分文字显示不全,建议横屏查看或者联系作者获取文档。)9 p8 n$ @4 E9 s. b" A7 }
11 b) |8 q- R( ~% y! e
对话框代理(Agents)
7 f$ O3 q2 Z- Y0 g% S0 ?•对话代理(DialogAgent)是处理与界面命令相关的一个接口CATDialogAgent,其中与人机交互有关的代理大致分为2类,选择代理(PathElementAgent)和指定代理(IndicationAgent),他们分别对应接口CATPathElementAgent和CATIndicationAgent。! F/ O- J7 U4 {$ }8 k+ R0 P& ?
1.对话框代理(CATDialogAgent):
  q4 _+ Z% d. x5 ^* \# PCATDialogAgent是主要类,该类可以用来定义一个Agent与命令对话框中对象的连接- j3 }( o) f/ D9 F
当一个事件发生时,这个Agent将被赋值.通常用于面板的交互和输入消息提示。
& Y, ]( G. |: k1 n. d" q2.选择代理(PathElementAgent):# \6 t8 a3 m+ a: {
如果文件中的对象显示在视图中,用户用左键选择它,这个对象可以作为后续工作的输入,如选择两个点绘制一条直线,选择三个点绘制一条折线等交互式选择。CATPathElementAgent代理主要处理这种与选择有关的操作。
" O8 @" x/ d: w- b0 \) W2 ]说明:
) F; J- |" x7 [•在catia中鼠标的每次选择对象都会产生路径,就是所选对象的位置路径,比如选中了某个点后会产生路径:Product.1\part.1\几何图形集\点.1。若选择的是实体表面可能会产生路径:part.1\零件几何体\凸台.1\面。这种路径未必会按照CATIA中的结构树的上下级关系产生。它是按照几何元素对象的拓扑关系产生的。
6 X1 p" I$ [- O4 W9 @- r& v) [•在选择后会自动产生路径,并将路径存放在CATPathElementAgent的实例对象中。所以我们在选择了零件中的一个几何对象后从选择代理实例追溯它的路径中包含的对象。 ( H' f" X; a8 x
•常用情形是在装配体中选择某个几何对象后获取几何对象所在的part名称,product名称,对象名称等。8 B+ C9 M6 _: M  O% a: G& t
选择代理的定义:! t! G, Y% Z) z/ y9 K  o% D% o
在.h文件中声明,
8 Z0 |4 E* G, L2 ]6 wclassCAADegCreatePlaneCmd:publicCATStateCommand/ @6 ^. @8 `: r5 {9 w
{2 Z' ?! _! ]- R- n% @4 H

; A& W9 |$ n# S$ r1 P2 h private:
6 T+ C& ~# W2 z* `) g) |   CATPathElementAgent*_daPathElement;0 h2 \' r7 S9 j! h* r6 }

3 w/ f6 Q# {6 K: B* p. p: y}; E3 l1 I3 e/ S/ \/ x* z! F4 O  j
在.cpp文件BuildGraph()中实例化,
. p! b4 g  ~- y( ?; G$ ]" p+ V% F2 \, X# [: ]: T2 G
voidCAADegCreatePlaneCmd::BuildGraph()
9 H: G, k6 ]. l8 s) T; C/ s{
" f$ ?4 p0 x3 r2 `- V ...
& f  ]9 d8 @6 O, D; F _daPathElement=newCATPathElementAgent("GetPoint");
! c9 \( n1 u* p  X  I# Y _daPathElement->AddElementType(IID_CATIGSMPoint);
( t+ b, r& x, w# D# ?9 S ...% s! _0 R* f8 U: b6 I, g3 q) Z
}0 H) |( E* o, I1 e, R/ g
注意:; i' B' }! w0 x& s" Z3 [: c
这里字符串“GetPoint”定义了一个选择代理在程序中的唯一识别符号。AddElementType方法的参数给定了选择代理只能识别CATIGSMPoint类型的点对象,其他对象一概不能识别。( z8 _' w( [, \9 @# {% c
AddElementType也可以通过字符串的方式给定,比如:: G8 i1 L: ~( m+ ^, S+ |
_daPathElement->AddElementType("CATIGSMPoint");/ N; t0 b/ \6 w
若可以选择的对象较多是可以采用字符串链表的形式给定,如下所示: % U& i( Z! ~7 P6 W8 \7 `# _
   CATListOfCATStringTypes;; |/ z! k7 x2 C$ m0 t9 v  f0 W
   Types.Append("CATPoint");# O, }$ G- }0 l! F# _' A2 R6 j
   Types.Append("CATINewHole");
% Q7 [; |$ g# Q   Types.Append("CATISketch");' _2 o$ Y/ f& u* ~" Q
   Types.Append("CATIGSMLineNormal");5 @7 i; D! x. j. U9 }( Z
   Types.Append("CATIGSMLinePtPt");
) z. E  @  `' h5 {2 s. {   _daPathElement->SetOrderedTypeList(Types);
$ u; V. n3 Q0 ?1 o( a7 H9 f
2 m& e5 ]1 B( O1 Y& R* Q- X3.指示代理(IndicationAgent); ~, @" w+ b  x) L9 s  m; ]  W$ ~
•指定代理处理指鼠标在屏幕上点击(空白处点击)或者移动的时候发生的事件。
2 K. c3 Z, H6 y- ?. l/ U( d! S•点击或者移动的过程中产生临时2D点。指定代理能够获取文件中不存在的临时2D点的坐标,这2D点是由用户在2D(如草图或工程图中)或3D视图(如装配视图)中屏幕上点击左键或移动鼠标产生的。 4 p' u' ]5 ]# S; V+ S2 b
•一个指定代理致力于怎么从2D屏幕所在的平面中获取2D点坐标,并将坐标转化在绝对坐标系。鼠标在屏幕上的点击是不确定的(只给定了2个坐标值),如果给定一个投影平面,那么在屏幕上的点击会根据视线方向(当前视角给定)投影到投影平面。(默认的投影平面就是与屏幕平行的平面,也可给指定代理提供一个平面)。4 i6 H$ y2 M- `% v$ _2 }; G6 H
指示代理的定义:, R$ h2 ~& B- m* _
在.h文件中声明,CATIndicationAgent *_daIndication;
, w- O3 S4 k' A6 C5 w9 h+ ^% m在.cpp文件BuildGraph()中实例化+ J3 t( B9 n; j6 m
_daIndication=newCATIndicationAgent("PointIndication");
5 C$ p" Z0 g. z% ?4 h& V) v) I0 c, g  d- m
代理的行为定义:
: D) P" B: P$ jvoidCATDialogAgent::SetBehavior(CATDlgEngBehavioriBehavior);
( M6 b4 f# S, y: |8 _- e% G( h" k% T" w8 \' j/ P% P
代理行为表现在代理接收通知消息后的反映,如定义一个平面类型的选择代理并设置代理行为为高亮提示,当鼠标移动到平面时即可看到该平面高亮。' z" r. L+ N5 }6 m1 m
_daIndicationP->SetBehavior(CATDlgEngAcceptOnPrevaluate|              CATDlgEngWithPrevaluation);6 t# W: p7 m+ C, k% C  Z( t! u
通用对话框代理的组成包括:3 w* U) a" ?' n9 V
行为模式:Thebehaviormode9 N) A: e% P5 w$ j+ c0 E
通告集:Thenotifierset/ b1 R4 {0 W  F
通知组合:Thenotificationpattern.
$ o; J2 L! T1 o# n% e3 |
! E) ?" M' j) `) A1 X如一个可重复/可撤销的代理行为设置如下
# X$ U- ~& R4 e! R+ k, ^# s8 MMyDialogAgent->SetBehavior(CATDlgEngRepeat|CATDlgEngWithUndo);
1 J; j) A' y) x. s$ Y2
3 `7 O+ q$ ]' C  m  q4 j状态(States)
# M* G( j# Q% h! P$ m; {对话框代理用于用户交互式输入,它通常需要匹配一个或者多个状态(States),而且代理的赋值也通常需要对状态进行校验。如下图通过两点绘制直线的交互操作状态图中,整个操作过程包括四个状态:初始状态,选择起点状态,选择重点状态,最终状态。因此“状态”可以理解为交互命令执行的每个步骤。+ q  J: |1 I+ k- N8 j/ v4 {
图片来自百库全书$ a6 B' s: z/ p0 b( N, P
状态的创建:; d3 A$ R! H: `
注意:) K9 c2 g: w) M# k9 U
  The GetInitialState methodforthefirststate8 F$ F, j( V( h1 k7 y) P$ x
CATDialogState*stStartState=GetInitialState("Start");/ h$ O; P  S% V# ]1 [0 ^& F6 W% {
The AddDialogState methodforthesecondstate! B1 z, d; ]: P# r) c
CATDialogState*stEndState=AddDialogState("Second");
- Y3 ]! H9 n/ q4 Z( O2 N/ G6 o" P4 ]其中 Start 和 Second为状态提示信息。! r" X# v0 Q& T1 g
不用创建初始状态和最终结束状态。
0 @2 l, c/ H& ^状态随着命令的删除而删除。8 x" I5 }- n6 i( R2 i/ C  X
其次将代理添加到状态中:
8 m4 q4 ~- w0 ]9 {1 B8 ^stStartState->AddDialogAgent(_daIndicationAgent);+ E( a: ?0 [1 v, C' H
注意:
2 s( B  c2 m: y3 o# x' `1.通常一个对话框代理只关联一个状态,但是可以在对话框的循环使用中管理几个状态,使用InitializeAcquisition 方法。
! ^' c! j1 X0 N. m& c, L7 q( W_daIndication->InitializeAcquisition();8 r" j9 L1 f  L! ~+ j7 V4 Z
2.另一方面,多个对话框代理可以关联一个状态。
! B4 Y7 [; _; c: G* cSourceState->AddDialogAgent(_peAgent);SourceState->AddDialogAgent(_daAgent);5 s+ J8 w# N1 h0 P
3, b, f0 t! Q4 D+ o0 g9 V
状态转移(Transitions)9 d0 F) |5 L5 {! ?/ u
状态转移连接着前后两个状态并触发相应指令(Actions),当代理被赋值时状态转移会触发,比如当用户选择一个点时状态转移会被触发,并且若返回值是true,那么状态转移函数就会执行。
6 H# ?! H$ c3 X8 `状态转移的定义:
) G( w& L6 b! `" t* y1 D- |2 `状态转移代码也是在BuildGraph中编写。
  J; l+ M3 g7 }! T) l! D" a: N6 |CATDialogTransition*pFirstTransition
+ Z( \% R" v: k# s3 d# \  o =AddTransition(SourceState,TargetState,...)
1 a( {! F8 Z( t: U6 l2 Z如利用两个点绘制一条直线的状态转移定义如下:- s- I6 q5 i3 x. ?7 H/ t0 d
CATDialogTransition*pFirstTransition
( z, x/ w( a" ^ =AddTransition(stStartState,stEndState,...);CATDialogTransition*pSecondTransition0 [; W. o% q( Q# m
=AddTransition(stEndState,NULL,...);( }: b$ f# V* Z
CATDialogTransition*pFirstTransition=AddTransition(stStartState,stEndState,
+ h+ z6 H! i; wAndCondition(IsOutputSetCondition(_daIndication),        Condition((ConditionMethod)&CAADegCreateLineCmd::CheckStartPoint)),  ...);CATDialogTransition*pSecondTransition=AddTransition(stEndState,NULL,
+ P$ S7 F8 f4 |6 S4 S AndCondition(IsOutputSetCondition(_daIndication),         Condition((ConditionMethod)&CAADegCreateLineCmd::CheckEndPoint)),          ...);9 x% v9 o" t/ ]- C1 U5 O5 }
状态转移的第一个参数为转移之前的状态,第二个参数为转移之后的状态,第三个参数为判断是否转移的条件,第四个参数为转移条件满足时执行的转移函数,这儿的转移函数为对输入点的检查。
& g3 X4 _* E- c- l8 X8 l45 n( T0 g$ `( P( O' H9 r4 n$ j& U( e
小结
0 o- e; f" T( b, @# c$ l2 X状态命令创建步骤:
, `/ x, u1 ]8 z声明和定义代理(CATPathElementAgent/CATIndicationAgent)
/ X, b* M* n! \+ v添加代理行为(SetBehavior)
/ |" t* {- O+ f创建状态(CATDialogState)6 m" u! X% @. f: K. [: \1 l
将代理添加到状态中(AddDialogAgent)
7 K) `, f9 F5 P7 p( b! H创建状态转移(Transitions)
5 R+ j% `4 d1 j9 R创建指令函数(Actions)
) B0 @! S4 _1 S- m9 H释放选择代理指针,命令结束的时候应该在析构函数或者取消中添加释放选择指针的代码。如下所示:% j, B8 \9 w5 n% L4 z( q- n8 c6 O
CAADegCreatePlaneCmd::~CAADegCreatePlaneCmd()
  i, K" }4 Q4 r- Y  M9 p3 c7 a{/ b' A  e0 V% x( h" e8 g2 c+ d
...6 y) b$ \. G: T( W0 E$ ]" a' L) ]
if(_daPathElement)_daPathElement->RequestDelayedDestruction();6 \0 b% P: g% @% y$ L9 d
_daPathElement=NULL;
( u& O& S' O8 N% [ ...* ?6 i( ]5 n9 X9 h0 R# p) a
核心代码格式如下:
5 J) R0 s% g8 L* o% d. b" |% W5 }voidCAADegCreateLineCmd::BuildGraph()
7 n: }6 c' J' n) H5 m{# O% l  f6 H" n% ~1 l
//代理& N- O+ F/ }3 M3 x
_daIndication=newCATIndicationAgent("PointIndication");
, b/ b5 Y% O# D  M& W' s+ B _daIndication->SetMathPlane(_ProjPlane);
) N  p, M7 y1 l //状态/ A1 \* C$ [- B, D2 K' z- e2 s
CATDialogState*stStartState=GetInitialState("stStartPointId");% b, J2 V! r: D( y& ?4 D+ j7 c
CATDialogState*stEndState=AddDialogState("stEndPointId");- N2 j- ]& S. j( h
stStartState->AddDialogAgent(_daIndication);% A, g7 k! {$ L( s: ~/ X! [+ J; ~& w
stEndState->AddDialogAgent(_daIndication);
- w9 V$ Q/ a- _) P //状态转移
, ?; o6 N- V$ P$ F9 L2 H CATDialogTransition*pFirstTransition=AddTransition(stStartState,stEndState,3 v4 C& ^/ Y" q) E% A/ S" K
                AndCondition(IsOutputSetCondition(_daIndication),5 L" B7 w# D8 w5 y
                 Condition((ConditionMethod)&CAADegCreateLineCmd::CheckStartPoint)),  7 P$ E& V3 `3 s* a8 S
   Action((ActionMethod)&CAADegCreateLineCmd::CreatePoint));
, f4 S* y$ j* M+ Y! }- N3 O5 A! M" e0 R/ V1 V2 \
CATDialogTransition*pSecondTransition=AddTransition(stEndState,NULL,% u+ J& g8 t, \; L( Z
    AndCondition(IsOutputSetCondition(_daIndication),
/ l  z" j( v- m  A                 Condition((ConditionMethod)&CAADegCreateLineCmd::CheckEndPoint)),   ! B5 t) I: v# X/ ?
Action((ActionMethod)&CAADegCreateLineCmd::CreateLine));
; F2 r$ Q; O% y}
" G# U' O3 O( F& s$ r. \+ n& z+ M$ y  M( m7 g
本文旨在说明状态命令的定义和编程过程,重点是基于状态机制,明确代理/状态/状态转移的定义。在了解基本概念的基础上,仍要结合百库全书中技术文章和案例熟悉其用法,欢迎大家在这块开发中提供相关意见和建议。/ b# _2 Y2 K, P/ N0 k

& t. Q$ _2 y6 C: E; L8 V
& A. v3 n& ^! o6 z' P- @1 j8 }
上海点团信息科技有限公司,承接UG NX,CATIA,CREO,Solidworks 等CAx软件,Teamcenter,3D Experience等PLM软件,工业4.0数字化软件的实施\二次开发\培训相关业务,详情QQ 939801026 Tel 18301858168 网址 doTeam.tech
回复

使用道具 举报

发表回复

您需要登录后才可以回帖 登录 | 注册

返回列表 本版积分规则

  • 发布新帖

  • 在线客服

  • 微信

  • 客户端

  • 返回顶部

  • x
    温馨提示

    本网站(plmhome.com)为PLM之家工业软件学习官网站

    展示的视频材料全部免费,需要高清和特殊技术支持请联系 QQ: 939801026

    PLM之家NX CAM二次开发专题模块培训报名开始啦

    我知道了