From efecbeb83f79809b3a03f15e9fc289cfaaf3a4f0 Mon Sep 17 00:00:00 2001 From: Nicogene Date: Mon, 3 Apr 2023 14:19:34 +0200 Subject: [PATCH 01/21] creo2urdf: complete porting to OTK --- src/creo2urdf/src/Creo2Urdf.cpp | 147 +++++++++--------- .../text/{ => usascii}/creo2urdf.txt | 0 src/creo2urdf/text/usascii/messages.txt | 32 ++++ src/creo2urdf/text/usascii/ui.txt | 4 + 4 files changed, 109 insertions(+), 74 deletions(-) rename src/creo2urdf/text/{ => usascii}/creo2urdf.txt (100%) create mode 100644 src/creo2urdf/text/usascii/messages.txt create mode 100644 src/creo2urdf/text/usascii/ui.txt diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index 3971ffb..2763204 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -38,6 +38,14 @@ else \ static char line [500]; +void printToMessageWindow(pfcSession_ptr session, std::stringstream& message) +{ + xstringsequence_ptr msg_sequence = xstringsequence::create(); + msg_sequence->append(xstring(message)); + + session->UIDisplayMessage("creo2urdf.txt", "DEBUG %0s", msg_sequence); +} + /*--------------------------------------------------------------------*\ ProAppData used while visiting Csys \*--------------------------------------------------------------------*/ @@ -51,23 +59,72 @@ typedef struct { ProError ProUtilCsysFind(ProMdl p_model, ProName csys_name, ProCsys *p_csys); -static uiCmdAccessState Creo2UrdfAccess(uiCmdAccessMode access_mode) -{ - auto model = pfcGetProESession()->GetCurrentModel(); - if (!model) { - return uiCmdAccessState::ACCESS_UNAVAILABLE; + + +class Creo2UrdfListerner : public pfcUICommandActionListener { +public: + void OnCommand() override { + pfcSession_ptr session_ptr = pfcGetProESession(); + + pfcModel_ptr model_ptr = session_ptr->GetCurrentModel(); + + xstring name = model_ptr->GetFullName(); + + pfcSolid_ptr solid_ptr = pfcSolid::cast(session_ptr->GetCurrentModel()); + pfcMassProperty_ptr massprop_ptr = solid_ptr->GetMassProperty(); + xreal mass = massprop_ptr->GetMass(); + + std::stringstream message; + message << "model name is " << name << " and weighs " << mass << std::endl; + + // Export stl of the model + auto partModels = session_ptr->ListModelsByType(pfcMDL_PART); + if (!partModels || partModels->getarraysize() == 0) { + message.clear(); + message << "There are no parts in the session" << std::endl; + printToMessageWindow(session_ptr, message); + return; + } + message.clear(); + message << "We have " << partModels->getarraysize() << " parts" << std::endl; + printToMessageWindow(session_ptr, message); + // Get all parts in the model + for (int i = 0; i < partModels->getarraysize(); i++) { + ProMdlName mdlname; + auto modelhdl = partModels->get(i);// = partModels->getl; How to transform it to ProModel? + auto name = modelhdl->GetFullName(); + modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create("CSYS"))); + } + + return; } - auto type = model->GetType(); - if (type != pfcMDL_PART && type != pfcMDL_ASSEMBLY) { - return uiCmdAccessState::ACCESS_UNAVAILABLE; +}; + + +class Creo2UrdfAccessListener : public pfcUICommandAccessListener { +public: + pfcCommandAccess OnCommandAccess(xbool AllowErrorMessages) override { + auto model = pfcGetProESession()->GetCurrentModel(); + if (!model) { + return pfcCommandAccess::pfcACCESS_AVAILABLE; + } + auto type = model->GetType(); + if (type != pfcMDL_PART && type != pfcMDL_ASSEMBLY) { + return pfcCommandAccess::pfcACCESS_UNAVAILABLE; + } + return pfcCommandAccess::pfcACCESS_AVAILABLE; + } - return uiCmdAccessState::ACCESS_AVAILABLE; +}; + + +static uiCmdAccessState Creo2UrdfAccess(uiCmdAccessMode access_mode) +{ + } static ProError status; -ProError Creo2Urdf(); - /*====================================================================*\ FUNCTION : user_initialize() PURPOSE : @@ -79,9 +136,12 @@ extern "C" int user_initialize( char *build, wchar_t errbuf[80]) { + auto session = pfcGetProESession(); + + auto cmd = session->UICreateCommand("Creo2Urdf", new Creo2UrdfListerner()); + cmd->AddActionListener(new Creo2UrdfAccessListener()); // To be checked it is odd + cmd->Designate("ui.txt", "Run Creo2Urdf", "Run Creo2Urdf", "Run Creo2Urdf"); uiCmdCmdId cmd_id; - status = ProCmdActionAdd("RunBug", (uiCmdCmdActFn)Creo2Urdf, uiProe2ndImmediate, Creo2UrdfAccess, PRO_B_TRUE, PRO_B_TRUE, &cmd_id); - status = ProMenubarmenuPushbuttonAdd("File", "Creo2Urdf", "-Run Creo2Urdf", "Run Creo2Urdf code", "File.psh_rename", PRO_B_TRUE, cmd_id, L"creo2urdf.txt"); return (0); } @@ -94,64 +154,3 @@ extern "C" void user_terminate() { } - -void printToMessageWindow(pfcSession_ptr session, std::stringstream & message) -{ - xstringsequence_ptr msg_sequence = xstringsequence::create(); - msg_sequence->append(xstring(message)); - - session->UIDisplayMessage("creo2urdf.txt", "DEBUG %0s", msg_sequence); -} - - -/*====================================================================*\ -FUNCTION : Creo2Urdf -PURPOSE : Execute the creo2urdf code. -\*====================================================================*/ -ProError Creo2Urdf() -{ - ProError status = PRO_TK_GENERAL_ERROR; - - pfcSession_ptr session_ptr = pfcGetProESession(); - - pfcModel_ptr model_ptr = session_ptr->GetCurrentModel(); - - xstring name = model_ptr->GetFullName(); - - pfcSolid_ptr solid_ptr = pfcSolid::cast(session_ptr->GetCurrentModel()); - pfcMassProperty_ptr massprop_ptr = solid_ptr->GetMassProperty(); - xreal mass = massprop_ptr->GetMass(); - - std::stringstream message; - message << "model name is " << name << " and weighs " << mass<ListModelsByType(pfcMDL_PART); - if (!partModels || partModels->getarraysize() == 0) { - message.clear(); - message << "There are no parts in the session"<getarraysize() <<" parts" << std::endl; - printToMessageWindow(session_ptr, message); - // Get all parts in the model - for (int i = 0; i < partModels->getarraysize(); i++) { - ProMdlName mdlname; - auto modelhdl = partModels->get(i);// = partModels->getl; How to transform it to ProModel? - auto name = modelhdl->GetFullName(); - modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create("CSYS"))); - } - - if(status != PRO_TK_NO_ERROR) - return status; - - return status; -} - - - - - diff --git a/src/creo2urdf/text/creo2urdf.txt b/src/creo2urdf/text/usascii/creo2urdf.txt similarity index 100% rename from src/creo2urdf/text/creo2urdf.txt rename to src/creo2urdf/text/usascii/creo2urdf.txt diff --git a/src/creo2urdf/text/usascii/messages.txt b/src/creo2urdf/text/usascii/messages.txt new file mode 100644 index 0000000..822bca6 --- /dev/null +++ b/src/creo2urdf/text/usascii/messages.txt @@ -0,0 +1,32 @@ +Welcome +Hello World! +# +# +%CPMyPrompt +This is a prompt +# +# +%CIMyInfo +This is an info +# +# +%CEMyError +This is an error +# +# +%CWMyWarning +This is a warning +# +# +%CCMyCritical +This is a critical +# +# +DefaultMsg +%0s +# +# +DefaultMsg Wstr +%0w +# +# \ No newline at end of file diff --git a/src/creo2urdf/text/usascii/ui.txt b/src/creo2urdf/text/usascii/ui.txt new file mode 100644 index 0000000..bbae6a8 --- /dev/null +++ b/src/creo2urdf/text/usascii/ui.txt @@ -0,0 +1,4 @@ +Run Creo2Urdf +Run Creo2Urdf +# +# From 14a5aeb769c1948437e948a2610b1f0a03899349 Mon Sep 17 00:00:00 2001 From: Nicogene Date: Mon, 3 Apr 2023 14:21:58 +0200 Subject: [PATCH 02/21] Cleanup --- src/creo2urdf/src/Creo2Urdf.cpp | 48 ++------------------------------- 1 file changed, 2 insertions(+), 46 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index 2763204..d0c07fb 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -13,30 +13,6 @@ #include #include -FILE* errlog_fp; - -#define PT_TEST_LOG(func,status, model_name,err) \ - if (err) \ -{ \ - printf (" LOG Error: %s\t%d\n", func, status); \ - fprintf (errlog_fp, " ciao icub-tech ho trovato un errore in aprire il modello corrente: %s\t%d\n", func, status); \ -} \ -else \ -{ \ - printf (" LOG %s\t%d\n", func, status); \ - fprintf (errlog_fp, " ciao icub-tech sono riuscito ad aprire il modello %s\t%d : %ls \n", func, status, model_name); \ -} - -#define PT_TEST_LOG_SUCC(func, model_name) \ - PT_TEST_LOG (func, status, model_name, status != PRO_TK_NO_ERROR) - -#define PTTestResfileWrite(str) \ -{\ - printf(str);\ - printf("\n");\ -} - -static char line [500]; void printToMessageWindow(pfcSession_ptr session, std::stringstream& message) { @@ -46,22 +22,8 @@ void printToMessageWindow(pfcSession_ptr session, std::stringstream& message) session->UIDisplayMessage("creo2urdf.txt", "DEBUG %0s", msg_sequence); } -/*--------------------------------------------------------------------*\ -ProAppData used while visiting Csys -\*--------------------------------------------------------------------*/ -typedef struct { - ProMdl model; - ProCsys p_csys; - ProName csys_name; - ProModelitem *items; -}UserCsysAppData ; - -ProError ProUtilCsysFind(ProMdl p_model, - ProName csys_name, ProCsys *p_csys); - - -class Creo2UrdfListerner : public pfcUICommandActionListener { +class Creo2UrdfActionListerner : public pfcUICommandActionListener { public: void OnCommand() override { pfcSession_ptr session_ptr = pfcGetProESession(); @@ -117,12 +79,6 @@ class Creo2UrdfAccessListener : public pfcUICommandAccessListener { } }; - -static uiCmdAccessState Creo2UrdfAccess(uiCmdAccessMode access_mode) -{ - -} - static ProError status; /*====================================================================*\ @@ -138,7 +94,7 @@ extern "C" int user_initialize( { auto session = pfcGetProESession(); - auto cmd = session->UICreateCommand("Creo2Urdf", new Creo2UrdfListerner()); + auto cmd = session->UICreateCommand("Creo2Urdf", new Creo2UrdfActionListerner()); cmd->AddActionListener(new Creo2UrdfAccessListener()); // To be checked it is odd cmd->Designate("ui.txt", "Run Creo2Urdf", "Run Creo2Urdf", "Run Creo2Urdf"); uiCmdCmdId cmd_id; From 96c8f270c3c6897324df8ab0ebd2b6d0d822f6e4 Mon Sep 17 00:00:00 2001 From: Nicogene Date: Mon, 3 Apr 2023 14:43:28 +0200 Subject: [PATCH 03/21] Creo2Urdf: get mass of single parts --- src/creo2urdf/src/Creo2Urdf.cpp | 24 ++++++++---------------- 1 file changed, 8 insertions(+), 16 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index d0c07fb..3386dc2 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -14,11 +14,11 @@ #include -void printToMessageWindow(pfcSession_ptr session, std::stringstream& message) +void printToMessageWindow(pfcSession_ptr session, std::string message) { xstringsequence_ptr msg_sequence = xstringsequence::create(); - msg_sequence->append(xstring(message)); - + msg_sequence->append(xstring(message.c_str())); + session->UIClearMessage(); session->UIDisplayMessage("creo2urdf.txt", "DEBUG %0s", msg_sequence); } @@ -30,30 +30,22 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { pfcModel_ptr model_ptr = session_ptr->GetCurrentModel(); - xstring name = model_ptr->GetFullName(); pfcSolid_ptr solid_ptr = pfcSolid::cast(session_ptr->GetCurrentModel()); - pfcMassProperty_ptr massprop_ptr = solid_ptr->GetMassProperty(); - xreal mass = massprop_ptr->GetMass(); - - std::stringstream message; - message << "model name is " << name << " and weighs " << mass << std::endl; - // Export stl of the model auto partModels = session_ptr->ListModelsByType(pfcMDL_PART); if (!partModels || partModels->getarraysize() == 0) { - message.clear(); - message << "There are no parts in the session" << std::endl; - printToMessageWindow(session_ptr, message); + printToMessageWindow(session_ptr, "There are no parts in the session"); return; } - message.clear(); - message << "We have " << partModels->getarraysize() << " parts" << std::endl; - printToMessageWindow(session_ptr, message); + //message << "We have " << partModels->getarraysize() << " parts" << std::endl; + printToMessageWindow(session_ptr, "We have " + to_string(partModels->getarraysize()) + " parts"); // Get all parts in the model for (int i = 0; i < partModels->getarraysize(); i++) { ProMdlName mdlname; auto modelhdl = partModels->get(i);// = partModels->getl; How to transform it to ProModel? + //message << "model name is " << modelhdl->GetFullName() << " and weighs " << pfcSolid::cast(modelhdl)->GetMassProperty()->GetMass()<< std::endl; + printToMessageWindow(session_ptr, "model name is " + std::string(modelhdl->GetFullName()) + " and weighs " + to_string(pfcSolid::cast(modelhdl)->GetMassProperty()->GetMass())); auto name = modelhdl->GetFullName(); modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create("CSYS"))); } From 94452283987aa716c9f687561c59b282fa6caf40 Mon Sep 17 00:00:00 2001 From: Nicogene Date: Mon, 3 Apr 2023 17:09:59 +0200 Subject: [PATCH 04/21] Creo2Urdf: print inertia tensor --- src/creo2urdf/src/Creo2Urdf.cpp | 49 ++++++++++++++++++++++++++++----- 1 file changed, 42 insertions(+), 7 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index 3386dc2..d018aa7 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -23,6 +23,7 @@ void printToMessageWindow(pfcSession_ptr session, std::string message) } + class Creo2UrdfActionListerner : public pfcUICommandActionListener { public: void OnCommand() override { @@ -30,24 +31,58 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { pfcModel_ptr model_ptr = session_ptr->GetCurrentModel(); - pfcSolid_ptr solid_ptr = pfcSolid::cast(session_ptr->GetCurrentModel()); + + + // TODO Principal units probably to be changed from MM to M before getting the model properties + //auto length_unit = solid_ptr->GetPrincipalUnits()->GetUnit(pfcUnitType::pfcUNIT_LENGTH); + // length_unit->Modify(pfcUnitConversionFactor::Create(0.001), length_unit->GetReferenceUnit()); // IT DOES NOT WORK + // Export stl of the model - auto partModels = session_ptr->ListModelsByType(pfcMDL_PART); + + auto asm_csys_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); + if (asm_csys_list->getarraysize() == 0) { + printToMessageWindow(session_ptr, "There are no CYS in the asm"); + return; + } + + // TODO We assume to have just one csys in the ASM + //asm_csys_list->get(0)-> + + auto partModels = session_ptr->ListModelsByType(pfcModelType::pfcMDL_PART); if (!partModels || partModels->getarraysize() == 0) { printToMessageWindow(session_ptr, "There are no parts in the session"); return; } - //message << "We have " << partModels->getarraysize() << " parts" << std::endl; printToMessageWindow(session_ptr, "We have " + to_string(partModels->getarraysize()) + " parts"); // Get all parts in the model for (int i = 0; i < partModels->getarraysize(); i++) { ProMdlName mdlname; - auto modelhdl = partModels->get(i);// = partModels->getl; How to transform it to ProModel? - //message << "model name is " << modelhdl->GetFullName() << " and weighs " << pfcSolid::cast(modelhdl)->GetMassProperty()->GetMass()<< std::endl; - printToMessageWindow(session_ptr, "model name is " + std::string(modelhdl->GetFullName()) + " and weighs " + to_string(pfcSolid::cast(modelhdl)->GetMassProperty()->GetMass())); + auto modelhdl = partModels->get(i); auto name = modelhdl->GetFullName(); - modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create("CSYS"))); + auto massProp = pfcSolid::cast(modelhdl)->GetMassProperty(); + auto com = massProp->GetGravityCenter(); + auto princAxis = massProp->GetPrincipalAxes(); + auto comInertia = massProp->GetCenterGravityInertiaTensor(); // TODO GetCoordSysInertia ? + + printToMessageWindow(session_ptr, "Model name is " + std::string(name) + " and weighs " + to_string(massProp->GetMass())); + printToMessageWindow(session_ptr, "Center of mass: x: " + to_string(com->get(0)) + " y: " + to_string(com->get(1)) + " z: "+ to_string(com->get(2))); + printToMessageWindow(session_ptr, "Inertia tensor:"); + printToMessageWindow(session_ptr, to_string(comInertia->get(0, 0)) + " " + to_string(comInertia->get(0, 1)) + " " + to_string(comInertia->get(0, 2))); + printToMessageWindow(session_ptr, to_string(comInertia->get(1, 0)) + " " + to_string(comInertia->get(1, 1)) + " " + to_string(comInertia->get(1, 2))); + printToMessageWindow(session_ptr, to_string(comInertia->get(2, 0)) + " " + to_string(comInertia->get(2, 1)) + " " + to_string(comInertia->get(2, 2))); + auto csys_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); + if (csys_list->getarraysize() == 0) { + printToMessageWindow(session_ptr, "There are no CYS in the part "+string(name)); + return; + } + + // TODO It doesn't work + auto axes_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_AXIS); + printToMessageWindow(session_ptr, "There are " + to_string(axes_list->getarraysize()) + " axes"); + + // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it + modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create(csys_list->get(0)->GetName()))); } return; From f5df99a3f4e815add6ec3671b7cc29c5cd01dd4c Mon Sep 17 00:00:00 2001 From: Mattia Fussi Date: Mon, 3 Apr 2023 17:45:55 +0200 Subject: [PATCH 05/21] set rotational constraint as axis to axis --- twolinks/2bars.asm.1 | Bin 202772 -> 196056 bytes twolinks/barlonger.prt.1 | Bin 210711 -> 212901 bytes 2 files changed, 0 insertions(+), 0 deletions(-) diff --git a/twolinks/2bars.asm.1 b/twolinks/2bars.asm.1 index a597ac5161119d04730e929721a20ee100f3b678..8b0a7b97674e39fd44144e692d2f2760e06ba02f 100644 GIT binary patch delta 23638 zcmb@u2|QHa`#65*+V>^p+9S9;9LwbWz2~ z$*W-GRAm29si?XrV-#FnT(JeTqOcz;1uea?kF!e%MqXY{4qHlfNJvQ@qoC-7ytcU1 zFj8_DMO9UKjH;5F3s#SAH&%<9ai@bUMovk^L2=_degIsZ+j=$>8M?5Ey`wQ&p6g#eShSLn|w}V&oj< zF>($nfAKlEDj=4)s;F)7vF!H<@YDAR^!GyeROJ+~X*A{-SHv<`#3VUaRhPdQT~*|q zFpAEqE*m3fF!K#`L6kYlsybq&X^(&o^4M5fbI@51+ed3gfrJdZi_TmSv0O>n33MpX#7?l2^ zc5-x8#;7Q$$!+@4lz|EHJw{blMGkpWR{o1nMcF}a!))12vAGP46iSNn*ir^F3MB^> zEScdDg_4Ub)|Al_t)jd!X^`P4tNz8WrlyK?E6OQimuPpvT`UyTjK)5$zWb30uA=Jf zj9q3lqmp-(b8@XTV)8(N78kJ<%*H4o>>9HL$`5PG(gLLwfLdqrA9k^|MO5hV&1Pfh zz^pqD6ZpKUCffM@T7}yM#%Xoi_=7GyH8cvzNMxa7Ic;I~N})s3h|B79OA{+oz1(G? z-K=t*3_S0@2E5iE%l_8I^!$(lil>tWTGrx^%^B#GWBLZMk`GOw?jIj@_@*tD1|10z~^kEy!KTB?An2~dWQVI z9Ix8Ay_zgqxc$ZpVwp|%+qF%F*)dab0j!p70!}>4G=glXan&hj4#su?`#Y!>#;joJ zN7Hxib=3*ji=Wj1%;I_sd4cB`>H`dNR}y&qqcV(^pmqw~XNJ!c<&5}&Ls!0j7qAvJ z;60~mNwac1!=aYfBSX(n*FWmMw&1+X6p_0eVm4ay>x&HklIujlK~!n{N#z@|pOs?Y zSJ`;Q&!|wmLhMg=`2=~!38u-5X#P`1)N53J5at{DOCal0y1IfG?pNddyT$5xfpJ=w z5PkkOBd9h_)PY$xLzsy{r9rOG!P%pn;_Gc$>K~{kdXEzm%uFiOKOM>gUB)lq#ocwq z{aDXc6xx1}rlA48NGH+lc}vH{Qzvs`Dd^bjU0ug|`PX{qt873@l%C4QUB0LvI*gMWC0N(^-MNGNP;ZF0jU6sF)p7GTJqEEe@2w?9@YLACdjQ zee}gkD8NuA`%b+hU*R2t4)3naat;Y-TjrS1f*$oW7dJU0?U|0zuUBf>5*3d%pyId11y-i##2zFjM8zhrKW{7YUWUa)0)-Ww;kV#XZ{zxfi9o^m~k`T=lq&rHQTO6ppQJt+`zRy+=Zjh0wyHgME57}Z{z9GG;oDc4T_hy+*<4}jdwdER9(OKGzx|Z(imNxvQL0FF zA?dgvRC}aq=G~mSy?T=hhj3bvWHUzsCSFuGntspOebaHkyXLl_bPGj&o|&eEq`h)X ztr3o_Z5B_}UPaSOPymwFGtHd|VwL8#vHW47jj~NBW-nc7gCx4Qmzje*3KV%aj?sQPbE+@hGP#c zM-(7J$%wJQL`8?q0@jp%r*n2N&E*Q0!@5sFY9{J}09MS;<(a`sn<=5944f{lP?|k} zB0c^-j{<)E^jRn7ocosfH5njIx7&xU8*^hKC?DvLsf1!!qhmgowCBbyFp1~IQA0Ol zq9}RfYYW1`Hk{yvv1_qSl)KK*o=%kDqD>K@wP1CM72ih`K0iqrrAwueOUzG{1g$}k zi$l#RiOmpYBGG~Ii^dA<`+?6G#6+R4YVB0B*JM5wtxHYkUfx8_`!&QldG>&?F4Z!o zDLPWH%0o1n?chTSh8D~?>Y6X%5v0#>60?XZh=>$75YSMJ253WOe%wlxJdCdm0&d%oS-b-Ult zgS$i*JFf}w+&URAolE6gh>l1fp~q0ER=0cN~xAGpo_orZq7z zG%du{D4Mb)f9sP>M|QVhOn&TufoCnnZOzm4 zT|mHV;l#6U5z&eM#ASen*Sl8Gkk2dPDK(?Q=X?r|{fc_=0^uv2l;gkw%QhaiD-U0f zgo<>naQi)9GBko{v_mG2sxm(kaE?I5e>8}ANMr;kQ8WTxx8$e|JnAP553oA}CX#h=#haHz60oNbUq*UuY^?>FM{xok=O1+ci1u$&A# zv~&2hFzIpzWC+tNPhpz`IUw4Y9qB-naD4&O8x3Y4Z84$dk#LyHYZR>+M(5|y0^NS` zW{#_VI^x;+K~A-~{6JL0g^Xy%A%#;r0M=OEx2L0-ZnaPpi9OvRDol6uKr`!I4CQW0 z6uoRzG^6(0eFEia)QYN;Oefqac*|vJ)PN|)7;yuhzWgMa3_Zr|w{g%O%BS?^r$IK_ z=ejie$`e#xO)$s%#qXg+j>>(LoE_pV3TCp&>QfH7W99Wh+;k_^$u8NLjH4nfLA(^F$?{32sxj2q z@i;M_-W1%?kGl$3Dfp(wV@+lEao(1tl5bnQ=HB{29(vxv8{tQNxh0wvHBRZF?r>w^ zmWVpm!`L>GJq1Yn#F^(uqf*Nf90CA4nqSxKiF?XCCZfL&j}oD;G(k^+^bXgfGV@5Q>0UJh#Ixvu)LKkjv;cOI+d0(0o{ zQNN>8!InFff>@JT%@)XKc#qYnI$8LKnX?N{ANVGFE-KEL1?V?<`CWD3R^mk&7c(9j z0cO@FfY}0Ilf?kE7Fr#9ECRtOx)%VCZq#QlNrhOm^FTBmN2j>FI3*Br{2Cqi#34h8 zpLX4Tw+fXd(qs&EV;oFqDIQ?xr-jU~jrX!gwP2QRi&3hc;AG#eM-O5w0N~;QJGb<{ z)-+eixws%zDr~06V~8gzD)iK|i~Kb0KokIc$_V9oEBaD46nHHX$ob41Y=}J^@MS$55?!y6}%iS#BfEnI<^Zs81oEkNRR0 z%x)9|%_R-(m~XC=sJ4sx<}yh9L=0UV0{6SP25jD@=&@a*-Kcli++CT%hWpSJ2ixk@ z4fhwZOSUaf%9Beg{H{$%ND0VOXn!sJIECH6TaIORGyQ*`g}$vv&9bw`M(xH$?M4-x zK<^*?xXNyBdS`N;SUV&wCHTDLk!iHnl$4;E^w_23ORye<68~>RxQh^q69vV*8wbz% zdIfpA08+J`DK$P5;p3o_8$hHm6s9?Q&FN4$Ec1fSt0>?hD1i7gjK5w&oTlx8G=-AFFSS1D}WR4$4 z1%~>$0H|@4zsos~01sauac^fYdp8$fZ+oQ2-q|J4!NbcwaI**L*aILV1kSnmZ;%3C z5&kVW03f1-V80lYfkI%T=#Fkn4n^kc&)@skTNUYF<+d9~ZCpL=y{X+;b# zw5ii4$Q$@Z33nxfJZwmocJTA_vUhd}bO4CpiY&a642r_au_zTV<{j)Ci^_u16|eaZ z=}aOv{PObB1SyTUs%mL~R9X615eZ zzk0TVH;LK`*k44yd$xiIug0OgP!_OzJW3Jy$caZCKplra#-p-P7+h=uDi0KOmRLT6 zc+SDc841yb_hyZP7lCn|K_;okPw z2G4I;NDv;mhT01cW}}XBNC{frTfS!RwkdTBhP84~nw-~Ab!AtnKW!kkK_LvRn}d>q zALpPLxurxKr>{;(-P+{bhJ}QGQPAb07&$$Et(*;lZBS_E23{IIl#62E4F9!;Ht^e^ zkjO9o>j*!$A=9rFxIyq677~TaucP+jKIfwJ>0p^Vs9?&KGbj{Jv;-x|#2t8M_iB0Z zSE-nnrEl|%ej4U{E3JxVIDm7lK{cVo_jkResddF}WfvY~o%%S-ZzO~*#P2j?UW?-K z_?J!|DZwatp-7Fh+f+Pw!X(wdb#8ZTEBTvfyKh_HRvYfFMID8HWjMoq#v!|FC1-gI zw^H7=Jf5W$6+{-zzxCU1=gME!ZCFL~uYFtX-@^Z*+3sWet#P{tE_;NMgtqM2HkoG| z`!9CU{agPw!PJIDbbngg&~J%J=FcN>k3tppkb8a^cfr$xV;_HeL9*idAhF_!1DNvj^i9S>dN^IZZL_T(oBE$c-FOvIF z!|?I|>NIROh!W-_EUj=)68`!JY{I}KfQuhQ;n1`)0oGP;8TFaq{VOOem?;Ynt)TQc zDQSVJ?XAGx)yu&x0LVW8OA%40QSxvK5v2?>|3HbuSZQz^+?K0Pr{wB}K8c!UsG2L~Do-y+ zd3GCPM*fX~5%>?w$3F;K3WdK}BG>=0l&+&_H&tYEfxk5{{7s(u;5PyR|HwB){qCn_ z{hI+E{e#(n|FE0`|FCfVgN2#mHwH%FKP0t`Y(D( z*p~`a+HxW6L-m&mYULU2a6rUAC_c>?ST;f^wNTg zf6tH<#y|8=7`N$nGyU$RrTGVa!ygQcz<*3S&isdk;U6qt{$OB1mTeUYEPv=Z|3NRr z`WpkAvToDY{OP6T`v<)l+aG!a{$tX|KL}dPKUgB!w^?Lii2bkdaqZYf$k2-ZgXO^= z3_P-9+bTnjKlFS4K|lHnvpGY?IQ~*`j`I(T^gmdb|13;ciwmU1MR0*CC=?&Ov>)69 zk8p#R5V(g2+(XZywi!b~JPPjS0VQ$mJYez$;W0i?558Jmh(?gVcl z-SdK=#8nZH8t&b>IS@p2JkpIwm4Zh^KpL1=6g-S{Y^uC23Z6n>f+$#zXs+A^nr&#_ z7(2sW3WklOD2OKjaP&#UR9!Ukc3N!19^4l(&~k%PcMqtAG$Z%?vW!v^j6iy$CBX&6 zsIz;)=q+jVAl;UHpF-~$BK(FA>vk4=Gtnwv&U z9{?qADO%taYQ*EnU|>pwp!NVLi!>4364DJ{IfxiaVFB80G!9sS!f@>I?P&ABUyg&; zzxsWV5!6~Cvv*Sk2TZgAlMvX(8Wh46oB)SW)EjYM!E8i+5J6bmY#MsS1}s51H)b1r z0dsz2QjoNEG*1)p(|TrHqqP; zVCt4E_=y8}4uO{tQaH--_w0r@36YMA+k_Y=Fad#)SaZVlPRN?sk~rZ6W^U6YJ8w=% zdY6CLgZ)K%=hufYZq*e;ZJ14R1KW@fOLrt#lkVWoZP?lqWWvdMfC?K#&S$^@+(!i;6KZMdJN=c-}VN*>V}*CXI~(;4$d-D8&mu*WC`qL{V%d`VY7QR^)LTtSw03J zkMOo#G?FY76yYnD_*oQ6S65_;SbYKSz61~%;ESH1BZa=X5xn4m#O{#?h@^B6WLcen zh0lPpI>slgk>bkA@m2ha6*GbGZROWSlTkHE5&QWN;VA$hCur0JVMRJ7J0@phTOaQ) zeZS_Ca7NoN>W$NO9OXnz)(YD{Dd9XnHB*7`TDr{1Y$g#F)GS%nIE!>5f3oI zC%r&=TE$-j*~~G#+iOAPSLeoBQ~I;k`2JdJ&b*^@vxk>8AN-mVIMK7qJlGm!LwKkF*?jL742XX^;A=|sKS%g)^%C4#9O4?x zc7fx6vI~|=0KItswe0ZQ4BX=c(1MbJVjO^x$9OWZ)+NwNSnf}$_f};}^>`{_tJEw1 zhAYY#CC&<0Uji|Epm<*SC=o$Z&jB`(nji0RNINz5*=vgBclYOqD%$2pvG@bd@8WNb>BZE^n1&kME`NjLvlnnf+GoWTphqcJE2 zRKQ04H4-HUpGyJvqatySQ;rN1FgVJ0A{@cBAI&}<>73isv^NEh2UQQ%Vgch;)aYEu*H?`q&0{{hDwt2mC+jqj z<=!`Jx21&cckCKE?em16LG+n;-ZpxxdE`!jZ^v~Xktyic#vzr{fmHcrWeMOU9z>}C zv=eyN2|VSB^Ca+T2Y^RKi{OR6u6W@oR=g+wbKp^Up>HdG6aWPhtAN!B{7N7SkGfpQ zCga7!FMdnhqF`S_b(SC?JTQwJGn& z7uNQC=aGP?*st#kQL=|lv+3r*@fn~9Dg-Xc03A@zVTMdlTaMnZn|}4e4w>hkqPK<< zVx9^3$BSlN_Ro;KutT>;tXKfxb71ATVR9{b$s+U1OZ_t_I5ZPHy2DUP9{=q1^HoDR zp`Mn{dApFOE^xtJPyvz(?QZdNl#i5(=CQ}oGYJ`9e`0(!)FWVG0(ny4=j-7Uh&(5t z1`PQOZpyr8Hj>);A%IF{SCo|SE{@AH_Z<;3(AVDs**kzAl$0)E3}0s45ayi>J!-18F`X^E&M|*UUA$j{cyLib5 z`}%u1|A#@D|6h*ke^LnR-2??-v1?#Hbe=!}c<_Ga9j_*wIvz1xkWGEZWoups*$46P zoc*wud$l$HtaX_-KW(6RTIFG>qAyWAdL5KLI}>bP9_uC*dY)|yz2josabf|93_e;4 zktQbrp*tm(Cze)t@xP=mt{~gl$E{y{jP<5Uo=-H$|C7-Ge?Ene3v_ zuaD8t8!R}=rf1H2Z#iO6g8DnQ#d`mA%!g+qm*YY)ei`WsL6=Q}!^Td@{Cw83Xfx7v z!BnT9Z9?VfA)8kbs0t|ev)j?rU#{+%wBU9-acdWFE|QHX z^xb_1r?pKUsZftx_vPz3PQ6Pbkm}>Ju)#+eo+mEeNz0Krb8+#3LMasB{IGQD{ywLp zWAiOmc@$X#r^hm!vdwS>^Xb9?s_jn})$&d(pLl$$CC7c2+_76qK0YlLPr}-#XsvdR zo{@2Te`wizJdNwt6Opg$qkC{3hBk*jmZi*eoi2->^gVl>B!dxYI^t|GQCp9@|7cwJ z_MC-riOZc3Q~ln^UL7HVBy``7d^9>VKJV=X-{yv&hvL5)QMS~slUD$n+nHhYL z`cQg)pmom|Z^Gmgqrq9?GK_*nZE+Sww`D4Nm(B<%i^@N!Cdgsg9SQlFgf{e3 zE9tHW%HwMT{Z9R|DgAdf<%BPLGd=HhIyKv0a%{zg=QWZ3RWkeDgRhJN_a)TtvILs~ z0;FzTRL^@j{rUFyrQ>#oh>%zFnc?ZK3yL>N2yLPlPs(_b9cPzcrVdx|Q%KOXXA0mS z$Az_iPtxeOKPkhHQ5kF;&Mr?|3%NRE)gv$P!a1Vz_KzEnKVFnN80ULL)_XvqZH|E` zCt@2K#Ma@sCzC$1VOO=c>1Obhwchp7se9EE4iIWJW_^iRe7XwWAQOrDBqE%}fDhfm2>{EQWMJLf5Q zX!R!R80kaR^J7ldZO(=A%t3flfyKkS6FsLatE+Xmb8OK)%klyfz~=-$&jYuH9C1zF z)Jn>+5;9jk?#Bhyr+UFMkl)e=y0GG_9Bv2QkHpN=1h+50!^^p(gg#TNQru1Do^AC# zD(&{2HmE|Qdg)A}M(~N0DxjkJ`ejA>uZH20Ue0M~C8x`>?|i>nxR-y&6)~PHa`%_F z)AC?uPSZa*7OxX_ndiM&N77qIQ40y@AL-a0wEy&RDcRQwM+3h{e7FpqYgKavI}Tmz zr^n7clw~8uB}dLG)_6TL(%Yk!amrU(Kg!mu;HJs`A^qDcJw_7k$DV3Y9S-(5PgzCDh&U45B7-TvFFKhtl}P0?baFN4b%hoZ&-&LMFe;-Am~9FF z;C@nyvC*_=2_U;T?*(tW|LCSju`d`+@b~ zN#^G3R_+IzT`yj-JJWM6cTa!vM;ZzFP9ZUDh>_-r6IV-dq~>(L=ToXwAS)1t!G$z~ zFg4`%`DW5*O~0E-;=u1$<6#}QZlb9Dyu(t(aJ&7IS#og%-D2F&lq`-x9S(fI;yS=@ z>%nqwM%}0J{LKi}s)t#2sg|!Ndqp4DyjcrXX?f?%W^~8B^7C-(Pt)a}Pg47H)z^U- z#DL^8vz&v6{YvP1 z(?=C9T^fC-_}~G~%Dm9EC&Tqo;V#ji&u%x;5H)l#HBVnG{g)T?PXf{##7~ms zmwZO8=KLY|2v?Ur#ut0q@eytA3=^Fuif)(qdP81goR`DSHA0hj@=aY_YWgAtKdbO* ze8!(#{6NS#dyhEW-fD;L^Qt!)KXJQeO{#3BkxYB`j)(nXMXduhv8C1?gE-cr>hE2N_iRp@Don$kEG5`C z@uz~c1G7Eag4oYhW*?ra3m?LX-v6F-p81ql5u@gN+pM#f&a7BsE8lq*c@n&Xbllx< zCavl}rC+;EDq07|)5a=g+R;_Bjm+~p6S)z=kPLPm*nxaWcK}ASEYvs7@#_$?BBWN2 zUnzWdKFjEhOn&(~5XXOH5uH;y8{DvSNo8#pw+SH>I_`eo)8X@^@Qah#5=O=H12Xl_ zrQ(gMFGAnf9?cQ!G&7TV@pe_9h&8S`E0jyq>G z(al3@Yzxw+7iN-O zw&n+~I#wMk^j&E2wPG5~lZ;GUN;HT}iriE_>`8FlR$fi!LNsnEADPh^QkTm6t=xE0 zhvs~2%MF@ex=#|JiG_K7K?Ov$A@tKU@!NxXzN#~?v*OPL6f3Kf^WQC z7FIo)swjHvadCL4QeC9!ZVpQW=M$C&9KiqPWi^wyw!@cjZ5>EBODbGunJ3w->P*1T zSSmv(CTZTukY5yfK+rI^$vQw=ob221uFO3W{SXPaQ7gMXsbtdBcU6CtqUlC^N@Jy0 z?ux}BvejVzG!F8rN;w>`lRV?2^WiS)>V#!vkmPPwZ`)PWF4N%C5_QKyZ_caMc4_5F zM!fi;I5+6J%BE^~YZsljXz$mC&d_4l`W|xhvel<`0I?3*G%tmet+;8P{-iGw@NN4B z@6Y575$^pq&odX(fSC8VWvS(tc{;05dzj9KdFdVIha#h_-u0I7mB$}gQ%Fkx+A#g2 zIB;#S;2Sosq|4&!lDCeA*C6(|=Zkm_%i{+&9B9>shN>FpXjbLP{q`Pxb)fZ~#VQp!b&^!4ljs$p;jj+Sb;VDn&M|S>ywy}7T?{%) zXkFD#O3El1h>bt7@0QR~_EGr3W9?4W>Ykg0YI-I6%;aqyhoMC)tZ69vxzW7LIzX8~ zy4^qnoiDZ}279anOibek>hp-#LMRL$xP}?d zOV!Nb-v4BY`(on+ceQWjvmd7Rp9}2wtVNP8Laz41hdXT6f%xl-Wi6?br2$@F*Gv{W zlZz9ycfXrC*MP2F97rvBftQjVl`^s;KkbEzaOBR{^|r`}&~drmGKS>7#VR3?jQX z7KWjDawCIIji&V?1Cq|-@#7}G^Gy0=wmgZk1H|eAkF<55S-<3b-YS3C=XC&L)u~iG zuVL6r9-7d|LUR$zTI(8z0%hWB1|EiH5NP?i5Pf=&=C$&j9>MxLXN5oEDl<4oHf|Iv-*qm1o=qPy zRvdgm4lC{{M@|-ToZrgY&4(lj{-h$E$|B<6Vqr_JVqaihg;6mv@l}6>+T^=%N%Z`= z>Ym04xy9VVeBxbuW&`afcgLV+1?7igiDO6V$bDGT^N=Bf=bgx;o45Vq57ZuwbK|ae z;}n1F9V9xCqy8Y7Sw7=rhV$e=m(=YBkK2&EeO2Qc8^a_`Jh^vLW+(tTAXO>`eT61R zLddHIiwms`5$nL|w@BJ3el}`%aon6E6lWzQJm|X))L@7Kq>DG-T2J4A)^`1DUH!18 zz2p|V2HA|uu$~dxu{0Q8Z#3S>>7nC-WPt8Vil*i*xtQL_5f|^06YGGx`|z{U+(I`( zr~21!i|SQ_D#*Q}uBI+AV#hNPs30h$y{&O*Y#g5Fo*+gjEZv)EOw$iWO@rLayCU zNjB+bka5r=WtT~q{?KwWpZbIr4eog*#xHSi*yu-Q#^8eT$Qy9_;5 z`vN(6=CKgJHAQ0jB5W2lp4BD8@pEMnmf^QllItb$u5V<5#9mkT^?Fr#Y6Y7OdM$gK zJDq&3MB+BAOnIrHu@0~hW18DOw(lQUbNb8Q+9ug&c`rT)lZByCje&5WYp$xNf5g#Js8AgLfWdy{+7i(NWZ zk-hn$aEDsH#6O9JS|#MrcYsa z^b@~MSa!z5$Fc>39~zH;7!!8-B_BA*K5Sh4;rxwnW}TshB#O^v-Rl5x_t?A~vMUL$ zs=A}!{lpOxTgT$p0oO&ER+bujQkiwXPUXB6na!MU>^#nHP*d+bvV#OC;uBYowB8sa zKoDUaNLhviA=jZ@S97nCfU@f1!%7bHSBP$$yFQ%n{bn}(qfGyk7~PTV0;$R2$a6zw zyYCNT5g*>i{$N3TlR)ksE}JV_J#6RMKw+xi+`zO{lb^>Kxx-Ui&?bjjhkoW>N$E%2 z#2CvApYo-N5lQ_%O`6GS6+59fno3F|Pi9K^ycf< zT^w5zIo|oHeH9rel^E$pFX^K7kB&EYV0umyu)Zju_HD3sy`Vi^tNBtgm zb;91&+Hmo)|G7_?q#tuu+AQoNd+N@V&gh9be_z4BBn|m@6q??7sUh;Nr7q+y-LcDV z;^FCaD^+%LSRyPxlh>L%M}E@?x#_&SNAM#RM|R&PQnW#DM^*B=)`3fI_&cQ$N_gDY zMw}RO;I>VCS5}qE<^4vHVI*>u%v$Q(+W7XQVog8@c9riVR3Y}3lMH?TskbOerq5bE z9wAiUbHCnra*TM2OxwF+GfMQF)V0;OtF5db%pXwFxTkYrNgtUAjcbr7+LiqJO`yrb zj+scA*MTNldQRG0eNt7OySqB>rnu)6+HpbMrnX@(nMR;@rDYsVC;MF-dFd0O>k%Z@HCMnK&;((XV-Pu0E(FOC^#WH#M|E^1`-%^9I(3?z%?^Jy zI%?{Zhcza<<*1rvKXT*d9=d1WsdeLsX?n$)0y#HCH zt|o^%A8)gym782*aR2G}2)Z?_H0JU2eVWBodzn_|ta}LuM|+Ok`Q+UvQ20ELRDZkw zd-YGDnv+FbFOD@vztn7;H2!8YHJO)p&C-DL+oxM_!CO!O;;g)BOUj+w-@X!T-Y!-DKV%_5U@!sNcC-8fPKDSF)0soj34 z#ytGaZU`!;Uk7xU{!Ul_o~`C|wv-x0cJXa0EkKqT^joRtK#=*|>apKStGik4p=R3& z;>0hVd89`C6ZD2oW0&IF+Bk{UJz@q=_y`$gpCF$Q@?l(%NXIMjHKm)6_d%=9d3Y|w z;>ak(OLMs=td=e!t6Yr)+Yp>c#Y%+5(e%LX_%Tk??pFImCa2S0Ro;Tt3~ky{i`uzf znmYL*!RFdbOQKjok(QHmeddV~sL~K%Cb%2A(*RZ0t<56oEGM!fZ?RXWQsx3Q0SWf3 z167fG*2Gt>TFn#f9g)Lt>6x*ifz*v08hu~8Wnvy`Me@<&HKM;7ay*PsD1h9cX8k1! zgGCxb&es#u-Ki={%_L6->0@D4zQ=QL5%O5K!}e|+o%_pvsU(=ic*q#v}768c6958jM@kRq?lEE@tHdb9VtwQJ>4L+k_zq8{!xf7GL? zHj&EpIgBa(LsEI~t(u1Z(jau#B~za-V+?y%k47v>vWc&@f5>8N-`kWYYaDNJu{^+5 z>p_W2AoYtlp97X0&z#aK<&GVr%uK71r}Ucgv3PeP+P3fP@!;aaM?byG%50PzyLyl$ zNTSOYR_ah~GpS04g1(z~Dy~7@(=E+oi#EPdR$)xZ?a2*0)=n=X$#81bijaF{!hMY! z&qUqp@}b%Kd}3=DE-8ZP&KVu|-hnqWv-3h+Z^$n~m7ZO;_l_CRGF-?fBJ-Xqo0P37 zMtbb9*xj7Rs=arHfF8H`*L+#f>f-~#DNJH5!&Zs) zYK0NUtYpeu8nZhw^xy@&!L-0Z>&)QO+{pij;?#d_T*BL+*k0#Csl@lOF_mXlQ3Pw@KjIKZFVzruX{O=Wv{~rYm=-(?CVLW_ai&ckK z`=*Lnl%Oz~7b9aY%^&WFJG0N?>KjuX$+^KB=R7;npV0=9$ld9w@;cy=N_1ZsYSAT+ z^*2^&?_QHxRf*EZoUcH3rLe9NQr?-y#iWpd)%iNYqe+GZqDj4ufX;BP=m4@OArf$F z!G*-xRpl3ZtCrh8Q=KnWzjBZEXljdUiHzizyc@yhondG-)q~j|=zO_)ztwbxD!JBg zBy;DyHuM_DEa4PV-Ek$|u1B_A+Apc3oW;t?#^HvY5$da4c?2zPV{*@92+_&pKtQv& z3kj4X2~(rh-<*V<$YEH8jL2*Yi9ZLUbgC9|S1lvRL4$-)WIn5tV4aEENHu{K$%-0< z$d5;e?*ACB}<82>NpU>l{L z1WwG0W;!PjXrOlfT0$dB=q(H7;A`J*Q=!A z?Gy0pH614;0;v~7w@$!`1e$22)Q$76)|dS2}aPW^taLT`j|sCM%0&x2%ip5@Q?_tp(hWybeqwKpctnW|HKPW_cN&g-jo`Dh;(LME2SV zBbZ<#au?Bh$$s9RO3EI6qqMt-r_6;DED z3++k^$+HPP9Nc|=V+VZsZ#&@j1_=E>+X1gW=opKcU3jV?U2!?^v3hytlVq1*ceB^9 zOYoNt@<`@`Rb_}L(9$5nc!7$;Qg`bGKk5Wo;HFOG`?%Y0)DXY=uCbvnt%soCpc;{O zV?AvXzx(ZbJB3ikgj`ot_3XpR&5c!VYYN=+=<#m!jFq-gj8dLTQ{xVufRA&FVi$ex zM~>Z%#On`!KZp3w-0ESErYAuVd@1FIRN3M@$!xX0n3@5f6>{)@tfoSLms7{@9b`Aw zILLk^b~)Iyw{CvV4%L!2i}ZTuZ2Lm{8$D%;smm-k8)c9Ka)u6KlXTut^I#L-NOu+< zih_oOs2WLK-8Dq^aGN~h@v!2byAz8%xW-)HD3+Q$wRcEZu^vDk-NcEI*-aytd?HL} zuK$d-hddA7I@23}U|zM@r`t6z+xbRhu3Oj9*dnzfNDVdYL2{^K?~fmd15dI-Jka8i zvY`MnZx-Rk63d!Moem+mxjP9dAQ3Mv&d!~L-Y(mUts2fF$w7KiDe-NDR(^C+ z?S&4(JRwkK9!)fAK+oC%5t);h{ML;}3OKK0eXKG9Q?;S1;tD38_id^G1 z_xBzOyj)mxWx+u*{EWnxZ}A1vUphvZVA2zQ+m%B|DeIeAKK%L%@3g*ZpCr8&f~gwZ1`LGdAn+lJ6&~61@5LQglA*!pOGjKqcb!KT$mk+=zopAi7qdj z=UoRFpRF1Z721Dhc3&A!cI#Y}Y3+Xh>nOZ~X0!0=|x5BfhA{l4cRMZW@4q0U4gxr?g-82Az~#=k~Ee!hsT z@h|7j1s3m!zW>xu0(HT4XRYsM%u1b|PVf13U=gOsk$rGC0_g+4v z8@&`cH&|6X>(Y(Baq##8uaGO`U^t_6v-)@(3EfOKTZ%MA-$5Qpgu7*Mt|3M7r)G^V z-9%U4Vb4~b#T2BvJ2b|iElIjJDb`Z<5UFuH^#tb4^W%_VJ#HF6J1i2SCpuV&`t?Sl z2&p#j>&7Z*LWl^JhES}U{aC2x?k0rVcNVX4mp#ix*Dv|b4UR)L-ukbWtP(ZVmQ5HJ6?0_nUN6o|CNZ3zjxCYAYJt{^+c}G;u zsU*@Xa`rs!4YT`}9#{CWh&x=m+`Lf3Z%2mbow9_p>FHmSKAKN0H?EZSkt6TiOd{gn z{V51T|Mh|5=JQ15K$FJGkqg(zUuM0GQqp50u1;8|Sl`eD?`n2_@r!9_CuKq&dA#Ho z>j1anSW+a*!`|N2BlXKM^zSgQ{}; zoA)@E_Xob)-S}{z%2>&k;utaV@e@f)o45O(UM`eA<6AjW1T_qgs7X&|IH}gY5aZo_BOH06T!uAX)~U81>L$#PvUNPh zDyJKOr96U4@~3Z&lS9NANqeS{d1$R{9x-~-%7HLDFY+uaFd1C2-*_e?X|p6l;7XRt z9u+0GJ<6NANI>Q*DJAOD zSqIvXG`}=EsWLCJA9NJ=@=67d@V{hj;Inj6c7KGU7mxngAOxV+V_@5d<&dxWS zdq+Avy?mr)WXhx$zIXbp99iNjy?+Mx{`~w}Sj28u2^WdxQF(^Y^!Oja;x5{U<&J0K zq`5!Pgzv=V3|_h6^y%VMu?wxiw^{>SX(_i#(P(Lr^UdmmUp^g2Gw?BL4M%(nQ&Y@@ z-AldlY;D5q_opGJL41xde3pRx=?!wKW`Uy!ARTn@{>zsypIIMpaS@d-S7W?&I);{# z)t_Z&dJ^z1(Y_%SwB*2t?6p8%P%xb068!>*=lZgl&0fHyu z_~mnf&#ryAhGpV7?p^cFyJp4rcoULQbvFtS7`XH;@>encmv7os6nYT9&w(#U7q4@`RqB@EA;+@KWgso+jVh_WO~Mb$>6K6 zTV$P^os!&_$iVp95AQ!We!yuH#(eVnK=PZ5_k}(UR^J#q$4cqpkZor5-aUowqWO(L z`79FbyrSGPh*zbQ=3dP|JpmbL82v0ur?;z)R$!Bq@bj`&qF1rjzS#H}!;se|;8Ppn zcvf*N0tEDE4XNOBZOv^i$;W1~a+(#>$j;=V05))Vr?p!G#b;g-}DNyd_7&61w1F~@BGsGVar(A2_78iXHi$&MwtXnR4&Z0gFrwoHu>eTvU&bVEs zwZ(rMe5WWHI;AL@>9up4c$5ZioJ(9gYZn3T;4IIs7p$2qoXSMXF`4m2hdyL%A5L?L z8i?$8>oYf#S$QFvhEq*;c3vY$M)o#QMvPv#9WnT9|GUJ!uz26@mJ2I&`%2MP8IQ4s zYn&>mO5KLDpUK}h{rjS_yVXH!>%N9)_2oC-uFrB$if%M#kumFZaWT3#q_Cxjm+)_v zO4xFPDGCCMlE9Mq4bx0FPM5=KeZFD_Zzjc;J{|OY;JxY>T-*4|z-gD6Tt?kF#%&W= zhJv+jlGS+~G3SpMI>J}ZYY?VL(K+`$j%-lan>}~ff%MgrbbEFv%yavhoCI-P@vdo& z`1XhdkrV+dwT@=IrrPKyV%9t^pGHn42c6Nij~ApG>L92fCNj9)L+Y&NIo9o|-}}S( zGbNxnbipRWe=YMvQ`Z+CblBFC?mL@jrT5|jQQjoFuA%r!V+G>ClD7i6dxSXrVxtQA z;J(tyexHL@QmdN#jB4D6?5le+^+OPU>6u=gw9qpyimZObe2OQ<*>Kmb(vd@QV~qmZVrozJbg@y6AVjQQE>zU`c=TUUmwyEIR+<>( z>3ygizl4{TwS==@=7nD)_MZ3#>lqmWzc&6yK8N?sN$p30$}kIYCybU^UUy!2>*AURv<4gx?|LPw&CzI5 zCfb&pZpUNNhScA(JC>-6IaYPl>jG>jG^nvhzvD{$rL6pg0^n<*Nv0X8{mjtH9LFHpHetlw#hVgm~Z!kY8 zE@G7O9uB{o8r-;3H3Gq$+<- z-T*HfzaDQdX)UGfQuQAVRSd9bZ?@~9Z+crvpJktCoTmD=%#Vb&p#x(GorSBl3Bxv= zbr#1Z0>}E>Q7uK3czXNB`3pY`CMf7`y-j_G7`Hl&Ln#TXD`Uh|sgB*p^Ndfyo3Y63 zk+%Nv*2O@vP3^2=wy%l*hQ4-OVal>fMwmzEeOQMY_QX|st))?HE$y_oxFlEfe3U&z zt=2MfxpBAJvD|~$_1@np`OkMZJ4~>r%zCTO2&~qmif{MVHeK|2C0Y1G68M`!G1J*F zugfIOrFG7MX?sb_QOBiI>k7v5+Q;NPO1{&|o-u;g2r&FoFY#8p%GZ!)v37tmeB6gI zy6n^PSiHr~rh^ckl5BJ$pwz!LmHLX9gJT`r%{oOYWTJC2&X&xpyc%1~P1PE|7%I;# z5Wu}hB&inMTo0+|+QjY6N6xhhZ6k@@HIT1riKPAWT{tjt0uw~x>sTqtDT!zk^%FFjOrN-M?cI-s3Drk(Pevx=D>XPpQ zeJhyXUs2MBqg9I%Q<(NC{L4D&uhQJC_~+sSz0DDnDY@K@scN`JG&p%V#Rl{Lx*i_5Pib<2{~x`MLWG!+0Y zXDMn4kqoOmj5Jusv{vAFist?19&_P*scz^%9FUDwgwdOBrr%8 zF?jI4EOG)sB~2(2xuCZu^o87v2$n%}eFDTWV**e_!p~}42-YT%$Q%@3c4&y426t&e zb1$fJ2+2B3|;0oZlJn$9*plpU*YaE`-hC0sA6JxP5 z@tVWqLwq*`H_NbYO%DeKo#n`|XF0mqc%g875}pqdkbr}1NF+kkG7oA{CK>h3kuArt z&G;w|7Xzvz0R78W3QAxr5>SJHWDLn197Y0rWJ4Yo+@TFP7(i^Y83qL*(Bz-y_TMS+ RW;h&*&*z$f>Ds`_{{nub_W=L^ delta 30374 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 diff --git a/twolinks/barlonger.prt.1 b/twolinks/barlonger.prt.1 index 4a6377fd4b9a8ab72f69ef00f662457a91db75cf..6820b449432fd732f360213137a952508d6e36b9 100644 GIT binary patch delta 22063 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 BHz@!B delta 20581 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 From 899e8d16579ce852121d1afc392e2a5209c33d87 Mon Sep 17 00:00:00 2001 From: Nicogene Date: Wed, 5 Apr 2023 10:45:09 +0200 Subject: [PATCH 06/21] Axis fix --- src/creo2urdf/src/Creo2Urdf.cpp | 22 +++++++++++++++------- 1 file changed, 15 insertions(+), 7 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index d018aa7..04ef545 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -58,11 +58,11 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { // Get all parts in the model for (int i = 0; i < partModels->getarraysize(); i++) { ProMdlName mdlname; - auto modelhdl = partModels->get(i); - auto name = modelhdl->GetFullName(); - auto massProp = pfcSolid::cast(modelhdl)->GetMassProperty(); - auto com = massProp->GetGravityCenter(); - auto princAxis = massProp->GetPrincipalAxes(); + auto modelhdl = partModels->get(i); + auto name = modelhdl->GetFullName(); + auto massProp = pfcSolid::cast(modelhdl)->GetMassProperty(); + auto com = massProp->GetGravityCenter(); + auto princAxis = massProp->GetPrincipalAxes(); auto comInertia = massProp->GetCenterGravityInertiaTensor(); // TODO GetCoordSysInertia ? printToMessageWindow(session_ptr, "Model name is " + std::string(name) + " and weighs " + to_string(massProp->GetMass())); @@ -71,15 +71,23 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { printToMessageWindow(session_ptr, to_string(comInertia->get(0, 0)) + " " + to_string(comInertia->get(0, 1)) + " " + to_string(comInertia->get(0, 2))); printToMessageWindow(session_ptr, to_string(comInertia->get(1, 0)) + " " + to_string(comInertia->get(1, 1)) + " " + to_string(comInertia->get(1, 2))); printToMessageWindow(session_ptr, to_string(comInertia->get(2, 0)) + " " + to_string(comInertia->get(2, 1)) + " " + to_string(comInertia->get(2, 2))); + auto csys_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); if (csys_list->getarraysize() == 0) { printToMessageWindow(session_ptr, "There are no CYS in the part "+string(name)); return; } - // TODO It doesn't work - auto axes_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_AXIS); + auto axes_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_AXIS); printToMessageWindow(session_ptr, "There are " + to_string(axes_list->getarraysize()) + " axes"); + if (axes_list->getarraysize() !=0 ) { + // FIXME I assume to have just one axis + auto axis = pfcAxis::cast(axes_list->get(0)); + printToMessageWindow(session_ptr, "The axis is called " + string(axis->GetName()) + " axes"); + auto surf = axis->GetSurf(); + //surf->get + + } // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create(csys_list->get(0)->GetName()))); From a1176af122afb69f90141654537ff293de2ed018 Mon Sep 17 00:00:00 2001 From: Mattia Fussi Date: Wed, 5 Apr 2023 12:19:12 +0200 Subject: [PATCH 07/21] Print start and end points of axis --- src/creo2urdf/src/Creo2Urdf.cpp | 23 +++++++++++++++++------ 1 file changed, 17 insertions(+), 6 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index 04ef545..7519d1c 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -7,12 +7,15 @@ #include #include #include + #include #include #include -#include #include +#include + +#include void printToMessageWindow(pfcSession_ptr session, std::string message) { @@ -57,14 +60,13 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { printToMessageWindow(session_ptr, "We have " + to_string(partModels->getarraysize()) + " parts"); // Get all parts in the model for (int i = 0; i < partModels->getarraysize(); i++) { - ProMdlName mdlname; auto modelhdl = partModels->get(i); auto name = modelhdl->GetFullName(); auto massProp = pfcSolid::cast(modelhdl)->GetMassProperty(); auto com = massProp->GetGravityCenter(); auto princAxis = massProp->GetPrincipalAxes(); auto comInertia = massProp->GetCenterGravityInertiaTensor(); // TODO GetCoordSysInertia ? - + printToMessageWindow(session_ptr, "Model name is " + std::string(name) + " and weighs " + to_string(massProp->GetMass())); printToMessageWindow(session_ptr, "Center of mass: x: " + to_string(com->get(0)) + " y: " + to_string(com->get(1)) + " z: "+ to_string(com->get(2))); printToMessageWindow(session_ptr, "Inertia tensor:"); @@ -84,9 +86,19 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { // FIXME I assume to have just one axis auto axis = pfcAxis::cast(axes_list->get(0)); printToMessageWindow(session_ptr, "The axis is called " + string(axis->GetName()) + " axes"); - auto surf = axis->GetSurf(); - //surf->get + auto surf = axis->GetSurf(); + + if (surf->GetType() == pfcSURFACE_CYLINDER) + { + auto xyz_points = surf->GetXYZExtents(); + + pfcPoint3D_ptr point = xyz_points->get(0); + printToMessageWindow(session_ptr, "Start point coords are (x, y, z) : (" + std::to_string(point->get(0)) + ", " + std::to_string(point->get(1)) + ", " + std::to_string(point->get(2)) + ")"); + + point = xyz_points->get(1); + printToMessageWindow(session_ptr, "End point coords are (x, y, z) : (" + std::to_string(point->get(0)) + ", " + std::to_string(point->get(1)) + ", " + std::to_string(point->get(2)) + ")"); + } } // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it @@ -132,7 +144,6 @@ extern "C" int user_initialize( auto cmd = session->UICreateCommand("Creo2Urdf", new Creo2UrdfActionListerner()); cmd->AddActionListener(new Creo2UrdfAccessListener()); // To be checked it is odd cmd->Designate("ui.txt", "Run Creo2Urdf", "Run Creo2Urdf", "Run Creo2Urdf"); - uiCmdCmdId cmd_id; return (0); } From 59e88f3d42c8d41d238f5d08c8af80c39c052c1b Mon Sep 17 00:00:00 2001 From: Mattia Fussi Date: Wed, 5 Apr 2023 12:21:58 +0200 Subject: [PATCH 08/21] Add example protk.dat --- src/creo2urdf/app/protk.dat | 7 +++++++ 1 file changed, 7 insertions(+) create mode 100644 src/creo2urdf/app/protk.dat diff --git a/src/creo2urdf/app/protk.dat b/src/creo2urdf/app/protk.dat new file mode 100644 index 0000000..e707e1a --- /dev/null +++ b/src/creo2urdf/app/protk.dat @@ -0,0 +1,7 @@ +NAME creo2urdf +STARTUP dll +EXEC_FILE \creo2urdf\build\x64-Release\bin\creo2urdf.dll +TEXT_DIR \creo2urdf\src\creo2urdf\text\ +DELAY_START true +ALLOW_STOP true +END From af65fe26e33cb4c78296fadf7d5420279f64f16d Mon Sep 17 00:00:00 2001 From: Mattia Fussi Date: Wed, 5 Apr 2023 16:07:47 +0200 Subject: [PATCH 09/21] Get transform between root and CSYS of parts --- src/creo2urdf/src/Creo2Urdf.cpp | 75 +++++++++++++++++++++++++++------ 1 file changed, 63 insertions(+), 12 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index 7519d1c..5eb7137 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -12,6 +12,7 @@ #include #include #include +#include #include @@ -58,6 +59,50 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { return; } printToMessageWindow(session_ptr, "We have " + to_string(partModels->getarraysize()) + " parts"); + + + auto csys_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); + if (csys_list->getarraysize() == 0) { + return; + } + + auto asm_component_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_FEATURE); + if (asm_component_list->getarraysize() == 0) { + printToMessageWindow(session_ptr, "There are no CYS in the asm"); + return; + } + + for (int i = 0; i < asm_component_list->getarraysize(); i++) + { + auto comp = asm_component_list->get(i); + + auto feat = pfcFeature::cast(comp); + + if (feat->GetFeatType() == pfcFeatureType::pfcFEATTYPE_COMPONENT) + { + auto feat_id = feat->GetId(); + xintsequence_ptr seq = xintsequence::create(); + + seq->append(feat_id); + + pfcComponentPath_ptr comp_path = pfcCreateComponentPath(pfcAssembly::cast(model_ptr), seq); + + auto transform = comp_path->GetTransform(xfalse); + + auto m = transform->GetMatrix(); + auto o = transform->GetOrigin(); + printToMessageWindow(session_ptr, "feat name: id: " + to_string(feat_id)); + + printToMessageWindow(session_ptr, "origin x: " + to_string(o->get(0)) + " y: " + to_string(o->get(1)) + " z: " + to_string(o->get(2))); + printToMessageWindow(session_ptr, "transform:"); + printToMessageWindow(session_ptr, to_string(m->get(0, 0)) + " " + to_string(m->get(0, 1)) + " " + to_string(m->get(0, 2))); + printToMessageWindow(session_ptr, to_string(m->get(1, 0)) + " " + to_string(m->get(1, 1)) + " " + to_string(m->get(1, 2))); + printToMessageWindow(session_ptr, to_string(m->get(2, 0)) + " " + to_string(m->get(2, 1)) + " " + to_string(m->get(2, 2))); + } + + } + + // Get all parts in the model for (int i = 0; i < partModels->getarraysize(); i++) { auto modelhdl = partModels->get(i); @@ -74,6 +119,7 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { printToMessageWindow(session_ptr, to_string(comInertia->get(1, 0)) + " " + to_string(comInertia->get(1, 1)) + " " + to_string(comInertia->get(1, 2))); printToMessageWindow(session_ptr, to_string(comInertia->get(2, 0)) + " " + to_string(comInertia->get(2, 1)) + " " + to_string(comInertia->get(2, 2))); + auto csys_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); if (csys_list->getarraysize() == 0) { printToMessageWindow(session_ptr, "There are no CYS in the part "+string(name)); @@ -83,22 +129,24 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { auto axes_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_AXIS); printToMessageWindow(session_ptr, "There are " + to_string(axes_list->getarraysize()) + " axes"); if (axes_list->getarraysize() !=0 ) { - // FIXME I assume to have just one axis - auto axis = pfcAxis::cast(axes_list->get(0)); - printToMessageWindow(session_ptr, "The axis is called " + string(axis->GetName()) + " axes"); + for (int i = 0; i < axes_list->getarraysize(); i++) + { + auto axis = pfcAxis::cast(axes_list->get(i)); + printToMessageWindow(session_ptr, "The axis is called " + string(axis->GetName()) + " axes"); - auto surf = axis->GetSurf(); + auto axis_data = wfcWAxis::cast(axis)->GetAxisData(); - if (surf->GetType() == pfcSURFACE_CYLINDER) - { - auto xyz_points = surf->GetXYZExtents(); + auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell + + // There are just two points in the array + pfcPoint3D_ptr point = axis_line->GetEnd1(); + + printToMessageWindow(session_ptr, "Start point coords of " + string(axis->GetName()) + " are (x, y, z) : (" + std::to_string(point->get(0)) + ", " + std::to_string(point->get(1)) + ", " + std::to_string(point->get(2)) + ")"); - pfcPoint3D_ptr point = xyz_points->get(0); - printToMessageWindow(session_ptr, "Start point coords are (x, y, z) : (" + std::to_string(point->get(0)) + ", " + std::to_string(point->get(1)) + ", " + std::to_string(point->get(2)) + ")"); - - point = xyz_points->get(1); - printToMessageWindow(session_ptr, "End point coords are (x, y, z) : (" + std::to_string(point->get(0)) + ", " + std::to_string(point->get(1)) + ", " + std::to_string(point->get(2)) + ")"); + point = axis_line->GetEnd2(); + printToMessageWindow(session_ptr, "End point coords of " + string(axis->GetName()) + " are (x, y, z) : (" + std::to_string(point->get(0)) + ", " + std::to_string(point->get(1)) + ", " + std::to_string(point->get(2)) + ")"); } + } // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it @@ -145,6 +193,9 @@ extern "C" int user_initialize( cmd->AddActionListener(new Creo2UrdfAccessListener()); // To be checked it is odd cmd->Designate("ui.txt", "Run Creo2Urdf", "Run Creo2Urdf", "Run Creo2Urdf"); + session->RibbonDefinitionfileLoad("tool.rbn"); + + return (0); } From eaf5b4122ab07df2e49c2e0d2c3771f2e0aa8d96 Mon Sep 17 00:00:00 2001 From: Mattia Fussi Date: Wed, 5 Apr 2023 16:29:08 +0200 Subject: [PATCH 10/21] compute unit vector for axis --- src/creo2urdf/src/Creo2Urdf.cpp | 35 ++++++++++++++++++++++++++++----- 1 file changed, 30 insertions(+), 5 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index 5eb7137..1533652 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -16,7 +16,9 @@ #include +#include #include +#include void printToMessageWindow(pfcSession_ptr session, std::string message) { @@ -27,6 +29,32 @@ void printToMessageWindow(pfcSession_ptr session, std::string message) } +std::array computeUnitVectorFromAxis(pfcCurveDescriptor_ptr axis_data) +{ + auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell + + // There are just two points in the array + pfcPoint3D_ptr pstart = axis_line->GetEnd1(); + pfcPoint3D_ptr pend = axis_line->GetEnd2(); + + std::array unit_vector; + + double module = sqrt(pow(pend->get(0) - pstart->get(0), 2) + + pow(pend->get(1) - pstart->get(1), 2) + + pow(pend->get(2) - pstart->get(2), 2)); + + if (module < 1e-9) + { + return unit_vector; + } + + unit_vector[0] = (pend->get(0) - pstart->get(0)) / module; + unit_vector[1] = (pend->get(1) - pstart->get(1)) / module; + unit_vector[2] = (pend->get(2) - pstart->get(2)) / module; + + return unit_vector; +} + class Creo2UrdfActionListerner : public pfcUICommandActionListener { public: @@ -138,13 +166,10 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell - // There are just two points in the array - pfcPoint3D_ptr point = axis_line->GetEnd1(); + auto unit = computeUnitVectorFromAxis(axis_line); - printToMessageWindow(session_ptr, "Start point coords of " + string(axis->GetName()) + " are (x, y, z) : (" + std::to_string(point->get(0)) + ", " + std::to_string(point->get(1)) + ", " + std::to_string(point->get(2)) + ")"); + printToMessageWindow(session_ptr, "unit vector of axis " + string(axis->GetName()) + " is : (" + std::to_string(unit[0]) + ", " + std::to_string(unit[1]) + ", " + std::to_string(unit[2]) + ")"); - point = axis_line->GetEnd2(); - printToMessageWindow(session_ptr, "End point coords of " + string(axis->GetName()) + " are (x, y, z) : (" + std::to_string(point->get(0)) + ", " + std::to_string(point->get(1)) + ", " + std::to_string(point->get(2)) + ")"); } } From 309675ebeba6357310b131401003f8e6b8b7c9bf Mon Sep 17 00:00:00 2001 From: Mattia Fussi Date: Thu, 6 Apr 2023 10:34:55 +0200 Subject: [PATCH 11/21] First attempt at adding idyntree --- CMakeLists.txt | 1 + src/creo2urdf/CMakeLists.txt | 3 ++- src/creo2urdf/src/Creo2Urdf.cpp | 3 +++ 3 files changed, 6 insertions(+), 1 deletion(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 5160c9d..06b9bfe 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -12,6 +12,7 @@ include(GNUInstallDirs) include(FeatureSummary) find_package(YCM 0.12 REQUIRED) +find_package(iDynTree REQUIRED) list(APPEND CMAKE_MODULE_PATH ${CMAKE_CURRENT_SOURCE_DIR}/cmake) # Control where libraries and executables are placed during the build. diff --git a/src/creo2urdf/CMakeLists.txt b/src/creo2urdf/CMakeLists.txt index f208110..1520f07 100644 --- a/src/creo2urdf/CMakeLists.txt +++ b/src/creo2urdf/CMakeLists.txt @@ -65,7 +65,8 @@ target_link_libraries(creo2urdf PUBLIC protk_dllmd_NU ws2_32 advapi32 mpr - netapi32) + netapi32 + iDynTree::idyntree-high-level) # FIXME all these win32 libraries that are used by protk dll have to be set as dependencies of the target ## FIXME diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index 1533652..a881ee3 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -19,6 +19,7 @@ #include #include #include +#include void printToMessageWindow(pfcSession_ptr session, std::string message) { @@ -72,6 +73,8 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { // Export stl of the model + iDynTree::Model idyn_model; + auto asm_csys_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); if (asm_csys_list->getarraysize() == 0) { printToMessageWindow(session_ptr, "There are no CYS in the asm"); From f61a0530648d3e01a930bb514d1254729959a8cc Mon Sep 17 00:00:00 2001 From: Nicogene Date: Fri, 7 Apr 2023 15:16:49 +0200 Subject: [PATCH 12/21] creo2urdf: refactor in order to consider everything a feature --- src/creo2urdf/src/Creo2Urdf.cpp | 113 +++++++++++++------------------- 1 file changed, 47 insertions(+), 66 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index a881ee3..a3359ec 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -13,12 +13,15 @@ #include #include #include +#include #include +#include #include #include #include +#include #include void printToMessageWindow(pfcSession_ptr session, std::string message) @@ -29,6 +32,11 @@ void printToMessageWindow(pfcSession_ptr session, std::string message) session->UIDisplayMessage("creo2urdf.txt", "DEBUG %0s", msg_sequence); } +// The key is id of the feature +std::map links_map; +// The key are id parent id child +std::map, iDynTree::IJointPtr> joints_map; + std::array computeUnitVectorFromAxis(pfcCurveDescriptor_ptr axis_data) { @@ -74,29 +82,7 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { // Export stl of the model iDynTree::Model idyn_model; - - auto asm_csys_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); - if (asm_csys_list->getarraysize() == 0) { - printToMessageWindow(session_ptr, "There are no CYS in the asm"); - return; - } - - // TODO We assume to have just one csys in the ASM - //asm_csys_list->get(0)-> - - auto partModels = session_ptr->ListModelsByType(pfcModelType::pfcMDL_PART); - if (!partModels || partModels->getarraysize() == 0) { - printToMessageWindow(session_ptr, "There are no parts in the session"); - return; - } - printToMessageWindow(session_ptr, "We have " + to_string(partModels->getarraysize()) + " parts"); - - - auto csys_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); - if (csys_list->getarraysize() == 0) { - return; - } - + auto asm_component_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_FEATURE); if (asm_component_list->getarraysize() == 0) { printToMessageWindow(session_ptr, "There are no CYS in the asm"); @@ -105,13 +91,15 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { for (int i = 0; i < asm_component_list->getarraysize(); i++) { + iDynTree::Link link; auto comp = asm_component_list->get(i); - auto feat = pfcFeature::cast(comp); + auto feat_id = feat->GetId(); if (feat->GetFeatType() == pfcFeatureType::pfcFEATTYPE_COMPONENT) { - auto feat_id = feat->GetId(); + + auto modelhdl = session_ptr->RetrieveModel(pfcComponentFeat::cast(feat)->GetModelDescr()); xintsequence_ptr seq = xintsequence::create(); seq->append(feat_id); @@ -129,56 +117,49 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { printToMessageWindow(session_ptr, to_string(m->get(0, 0)) + " " + to_string(m->get(0, 1)) + " " + to_string(m->get(0, 2))); printToMessageWindow(session_ptr, to_string(m->get(1, 0)) + " " + to_string(m->get(1, 1)) + " " + to_string(m->get(1, 2))); printToMessageWindow(session_ptr, to_string(m->get(2, 0)) + " " + to_string(m->get(2, 1)) + " " + to_string(m->get(2, 2))); - } - } + auto name = modelhdl->GetFullName(); + auto massProp = pfcSolid::cast(modelhdl)->GetMassProperty(); + auto com = massProp->GetGravityCenter(); + auto comInertia = massProp->GetCenterGravityInertiaTensor(); // TODO GetCoordSysInertia ? - - // Get all parts in the model - for (int i = 0; i < partModels->getarraysize(); i++) { - auto modelhdl = partModels->get(i); - auto name = modelhdl->GetFullName(); - auto massProp = pfcSolid::cast(modelhdl)->GetMassProperty(); - auto com = massProp->GetGravityCenter(); - auto princAxis = massProp->GetPrincipalAxes(); - auto comInertia = massProp->GetCenterGravityInertiaTensor(); // TODO GetCoordSysInertia ? - - printToMessageWindow(session_ptr, "Model name is " + std::string(name) + " and weighs " + to_string(massProp->GetMass())); - printToMessageWindow(session_ptr, "Center of mass: x: " + to_string(com->get(0)) + " y: " + to_string(com->get(1)) + " z: "+ to_string(com->get(2))); - printToMessageWindow(session_ptr, "Inertia tensor:"); - printToMessageWindow(session_ptr, to_string(comInertia->get(0, 0)) + " " + to_string(comInertia->get(0, 1)) + " " + to_string(comInertia->get(0, 2))); - printToMessageWindow(session_ptr, to_string(comInertia->get(1, 0)) + " " + to_string(comInertia->get(1, 1)) + " " + to_string(comInertia->get(1, 2))); - printToMessageWindow(session_ptr, to_string(comInertia->get(2, 0)) + " " + to_string(comInertia->get(2, 1)) + " " + to_string(comInertia->get(2, 2))); - - - auto csys_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); - if (csys_list->getarraysize() == 0) { - printToMessageWindow(session_ptr, "There are no CYS in the part "+string(name)); - return; - } + printToMessageWindow(session_ptr, "Model name is " + std::string(name) + " and weighs " + to_string(massProp->GetMass())); + printToMessageWindow(session_ptr, "Center of mass: x: " + to_string(com->get(0)) + " y: " + to_string(com->get(1)) + " z: " + to_string(com->get(2))); + printToMessageWindow(session_ptr, "Inertia tensor:"); + printToMessageWindow(session_ptr, to_string(comInertia->get(0, 0)) + " " + to_string(comInertia->get(0, 1)) + " " + to_string(comInertia->get(0, 2))); + printToMessageWindow(session_ptr, to_string(comInertia->get(1, 0)) + " " + to_string(comInertia->get(1, 1)) + " " + to_string(comInertia->get(1, 2))); + printToMessageWindow(session_ptr, to_string(comInertia->get(2, 0)) + " " + to_string(comInertia->get(2, 1)) + " " + to_string(comInertia->get(2, 2))); - auto axes_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_AXIS); - printToMessageWindow(session_ptr, "There are " + to_string(axes_list->getarraysize()) + " axes"); - if (axes_list->getarraysize() !=0 ) { - for (int i = 0; i < axes_list->getarraysize(); i++) - { - auto axis = pfcAxis::cast(axes_list->get(i)); - printToMessageWindow(session_ptr, "The axis is called " + string(axis->GetName()) + " axes"); - auto axis_data = wfcWAxis::cast(axis)->GetAxisData(); + auto csys_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); + if (csys_list->getarraysize() == 0) { + printToMessageWindow(session_ptr, "There are no CYS in the part " + string(name)); + return; + } - auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell + auto axes_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_AXIS); + printToMessageWindow(session_ptr, "There are " + to_string(axes_list->getarraysize()) + " axes"); + if (axes_list->getarraysize() != 0) { + for (int i = 0; i < axes_list->getarraysize(); i++) + { + auto axis = pfcAxis::cast(axes_list->get(i)); + printToMessageWindow(session_ptr, "The axis is called " + string(axis->GetName()) + " axes"); - auto unit = computeUnitVectorFromAxis(axis_line); + auto axis_data = wfcWAxis::cast(axis)->GetAxisData(); - printToMessageWindow(session_ptr, "unit vector of axis " + string(axis->GetName()) + " is : (" + std::to_string(unit[0]) + ", " + std::to_string(unit[1]) + ", " + std::to_string(unit[2]) + ")"); + auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell - } + auto unit = computeUnitVectorFromAxis(axis_line); - } + printToMessageWindow(session_ptr, "unit vector of axis " + string(axis->GetName()) + " is : (" + std::to_string(unit[0]) + ", " + std::to_string(unit[1]) + ", " + std::to_string(unit[2]) + ")"); - // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it - modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create(csys_list->get(0)->GetName()))); + } + + } + // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it + modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create(csys_list->get(0)->GetName()))); + links_map[feat_id] = link; + } } return; @@ -221,7 +202,7 @@ extern "C" int user_initialize( cmd->AddActionListener(new Creo2UrdfAccessListener()); // To be checked it is odd cmd->Designate("ui.txt", "Run Creo2Urdf", "Run Creo2Urdf", "Run Creo2Urdf"); - session->RibbonDefinitionfileLoad("tool.rbn"); + //session->RibbonDefinitionfileLoad("tool.rbn"); return (0); From 6fdbce29ee2c9bf3129da0bfb6d07f420b819bde Mon Sep 17 00:00:00 2001 From: Nicogene Date: Fri, 7 Apr 2023 16:43:07 +0200 Subject: [PATCH 13/21] creo2urdf: create first idyntree model --- src/creo2urdf/CMakeLists.txt | 1 + src/creo2urdf/src/Creo2Urdf.cpp | 98 +++++++++++++++++++++++++++------ 2 files changed, 82 insertions(+), 17 deletions(-) diff --git a/src/creo2urdf/CMakeLists.txt b/src/creo2urdf/CMakeLists.txt index 1520f07..cf09759 100644 --- a/src/creo2urdf/CMakeLists.txt +++ b/src/creo2urdf/CMakeLists.txt @@ -66,6 +66,7 @@ target_link_libraries(creo2urdf PUBLIC protk_dllmd_NU advapi32 mpr netapi32 + iDynTree::idyntree-modelio iDynTree::idyntree-high-level) # FIXME all these win32 libraries that are used by protk dll have to be set as dependencies of the target diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index a3359ec..118acb2 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -23,6 +23,8 @@ #include #include #include +#include +#include void printToMessageWindow(pfcSession_ptr session, std::string message) { @@ -65,13 +67,44 @@ std::array computeUnitVectorFromAxis(pfcCurveDescriptor_ptr axis_data } + +iDynTree::SpatialInertia fromCreo(pfcMassProperty_ptr mass_prop) { + + + auto com = mass_prop->GetGravityCenter(); + auto inertia_tensor = mass_prop->GetCenterGravityInertiaTensor(); + iDynTree::RotationalInertiaRaw idyn_inertia_tensor = iDynTree::RotationalInertiaRaw::Zero(); + for (int i_row = 0; i_row < idyn_inertia_tensor.rows(); i_row++) { + for (int j_col = 0; j_col < idyn_inertia_tensor.cols(); j_col++) { + idyn_inertia_tensor.setVal(i_row, j_col, inertia_tensor->get(i_row, j_col)); + } + } + + iDynTree::SpatialInertia sp_inertia(mass_prop->GetMass(), + { com->get(0), com->get(1), com->get(2) }, + idyn_inertia_tensor); + return sp_inertia; + +} + +iDynTree::Transform fromCreo(pfcTransform3D_ptr creo_trf) { + iDynTree::Transform idyn_trf; + auto o = creo_trf->GetOrigin(); + auto m = creo_trf->GetMatrix(); + idyn_trf.setPosition({ o->get(0), o->get(1), o->get(2) }); + idyn_trf.setRotation({ m->get(0,0), m->get(0,1), m->get(0,2), + m->get(1,0), m->get(1,1), m->get(1,2), + m->get(2,0), m->get(2,1), m->get(2,2) }); + return idyn_trf; +} + class Creo2UrdfActionListerner : public pfcUICommandActionListener { public: void OnCommand() override { - pfcSession_ptr session_ptr = pfcGetProESession(); + //iDynTree::ModelExporter mdl_exporter; + pfcSession_ptr session_ptr = pfcGetProESession(); pfcModel_ptr model_ptr = session_ptr->GetCurrentModel(); - pfcSolid_ptr solid_ptr = pfcSolid::cast(session_ptr->GetCurrentModel()); @@ -82,6 +115,7 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { // Export stl of the model iDynTree::Model idyn_model; + //mdl_exporter.init(idyn_model); auto asm_component_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_FEATURE); if (asm_component_list->getarraysize() == 0) { @@ -89,6 +123,8 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { return; } + std::string prevLinkName = ""; + for (int i = 0; i < asm_component_list->getarraysize(); i++) { iDynTree::Link link; @@ -119,11 +155,11 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { printToMessageWindow(session_ptr, to_string(m->get(2, 0)) + " " + to_string(m->get(2, 1)) + " " + to_string(m->get(2, 2))); auto name = modelhdl->GetFullName(); - auto massProp = pfcSolid::cast(modelhdl)->GetMassProperty(); - auto com = massProp->GetGravityCenter(); - auto comInertia = massProp->GetCenterGravityInertiaTensor(); // TODO GetCoordSysInertia ? + auto mass_prop = pfcSolid::cast(modelhdl)->GetMassProperty(); + auto com = mass_prop->GetGravityCenter(); + auto comInertia = mass_prop->GetCenterGravityInertiaTensor(); // TODO GetCoordSysInertia ? - printToMessageWindow(session_ptr, "Model name is " + std::string(name) + " and weighs " + to_string(massProp->GetMass())); + printToMessageWindow(session_ptr, "Model name is " + std::string(name) + " and weighs " + to_string(mass_prop->GetMass())); printToMessageWindow(session_ptr, "Center of mass: x: " + to_string(com->get(0)) + " y: " + to_string(com->get(1)) + " z: " + to_string(com->get(2))); printToMessageWindow(session_ptr, "Inertia tensor:"); printToMessageWindow(session_ptr, to_string(comInertia->get(0, 0)) + " " + to_string(comInertia->get(0, 1)) + " " + to_string(comInertia->get(0, 2))); @@ -139,28 +175,56 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { auto axes_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_AXIS); printToMessageWindow(session_ptr, "There are " + to_string(axes_list->getarraysize()) + " axes"); - if (axes_list->getarraysize() != 0) { - for (int i = 0; i < axes_list->getarraysize(); i++) - { - auto axis = pfcAxis::cast(axes_list->get(i)); - printToMessageWindow(session_ptr, "The axis is called " + string(axis->GetName()) + " axes"); + if (axes_list->getarraysize() == 0) { + printToMessageWindow(session_ptr, "There are no AXIS in the part " + string(name)); + return; + } - auto axis_data = wfcWAxis::cast(axis)->GetAxisData(); + // TODO We assume we have 1 axis and it is the one of the joint + auto axis = pfcAxis::cast(axes_list->get(0)); + printToMessageWindow(session_ptr, "The axis is called " + string(axis->GetName()) + " axes"); - auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell + auto axis_data = wfcWAxis::cast(axis)->GetAxisData(); - auto unit = computeUnitVectorFromAxis(axis_line); + auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell - printToMessageWindow(session_ptr, "unit vector of axis " + string(axis->GetName()) + " is : (" + std::to_string(unit[0]) + ", " + std::to_string(unit[1]) + ", " + std::to_string(unit[2]) + ")"); + auto unit = computeUnitVectorFromAxis(axis_line); - } + printToMessageWindow(session_ptr, "unit vector of axis " + string(axis->GetName()) + " is : (" + std::to_string(unit[0]) + ", " + std::to_string(unit[1]) + ", " + std::to_string(unit[2]) + ")"); - } // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create(csys_list->get(0)->GetName()))); + + + link.setInertia(fromCreo(mass_prop)); + if (i == asm_component_list->getarraysize()-1) { // TODO This is valid only for twobars + printToMessageWindow(session_ptr, "I AM ADDING A JOINT!"); + iDynTree::RevoluteJoint joint(fromCreo(transform), { {unit[0], unit[1], unit[2]}, + { o->get(0), o->get(1), o->get(2)} }); + + if (idyn_model.addJointAndLink(prevLinkName, prevLinkName + "--" + string(name), &joint, string(name), link) == iDynTree::JOINT_INVALID_INDEX) { + printToMessageWindow(session_ptr, "FAILED TO ADD JOINT!"); + return; + } + } + else { + prevLinkName = string(name); + idyn_model.addLink(string(name), link); + + } + //idyn_model.addAdditionalFrameToLink(string(name), string(name) + "_" + string(csys_list->get(0)->GetName()), fromCreo(transform)); TODO when we have an additional frame to add + links_map[feat_id] = link; } } + printToMessageWindow(session_ptr, "idynModel " + idyn_model.toString()); + + + + + //if (!mdl_exporter.exportModelToFile("model.urdf")) { + // printToMessageWindow(session_ptr, "Error exporting the urdf"); + //} return; } From 547163e8e7c48e332a3875fce8b0e60882a98f01 Mon Sep 17 00:00:00 2001 From: Mattia Fussi Date: Fri, 7 Apr 2023 17:52:52 +0200 Subject: [PATCH 14/21] Add small quality of life improvements and clean up print to msg winwo --- src/creo2urdf/src/Creo2Urdf.cpp | 203 +++++++++++------------ src/creo2urdf/text/usascii/creo2urdf.txt | 14 +- 2 files changed, 106 insertions(+), 111 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index 118acb2..e6f9dab 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -15,7 +15,7 @@ #include #include -#include +#include #include #include @@ -26,20 +26,33 @@ #include #include -void printToMessageWindow(pfcSession_ptr session, std::string message) +constexpr double epsilon = 1e-12; + +enum class c2uLogLevel +{ + INFO = 0, + WARN +}; + +std::map log_level_key = { + {c2uLogLevel::INFO, "c2uINFO"}, + {c2uLogLevel::WARN, "c2uWARN"} +}; + +void printToMessageWindow(pfcSession_ptr session, std::string message, c2uLogLevel log_level = c2uLogLevel::INFO) { xstringsequence_ptr msg_sequence = xstringsequence::create(); msg_sequence->append(xstring(message.c_str())); session->UIClearMessage(); - session->UIDisplayMessage("creo2urdf.txt", "DEBUG %0s", msg_sequence); + session->UIDisplayMessage("creo2urdf.txt", log_level_key[log_level].c_str(), msg_sequence); } + // The key is id of the feature std::map links_map; // The key are id parent id child std::map, iDynTree::IJointPtr> joints_map; - std::array computeUnitVectorFromAxis(pfcCurveDescriptor_ptr axis_data) { auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell @@ -51,10 +64,10 @@ std::array computeUnitVectorFromAxis(pfcCurveDescriptor_ptr axis_data std::array unit_vector; double module = sqrt(pow(pend->get(0) - pstart->get(0), 2) + - pow(pend->get(1) - pstart->get(1), 2) + - pow(pend->get(2) - pstart->get(2), 2)); + pow(pend->get(1) - pstart->get(1), 2) + + pow(pend->get(2) - pstart->get(2), 2)); - if (module < 1e-9) + if (module < epsilon) { return unit_vector; } @@ -66,11 +79,7 @@ std::array computeUnitVectorFromAxis(pfcCurveDescriptor_ptr axis_data return unit_vector; } - - iDynTree::SpatialInertia fromCreo(pfcMassProperty_ptr mass_prop) { - - auto com = mass_prop->GetGravityCenter(); auto inertia_tensor = mass_prop->GetCenterGravityInertiaTensor(); iDynTree::RotationalInertiaRaw idyn_inertia_tensor = iDynTree::RotationalInertiaRaw::Zero(); @@ -79,12 +88,11 @@ iDynTree::SpatialInertia fromCreo(pfcMassProperty_ptr mass_prop) { idyn_inertia_tensor.setVal(i_row, j_col, inertia_tensor->get(i_row, j_col)); } } - + iDynTree::SpatialInertia sp_inertia(mass_prop->GetMass(), - { com->get(0), com->get(1), com->get(2) }, - idyn_inertia_tensor); + { com->get(0), com->get(1), com->get(2) }, + idyn_inertia_tensor); return sp_inertia; - } iDynTree::Transform fromCreo(pfcTransform3D_ptr creo_trf) { @@ -101,25 +109,23 @@ iDynTree::Transform fromCreo(pfcTransform3D_ptr creo_trf) { class Creo2UrdfActionListerner : public pfcUICommandActionListener { public: void OnCommand() override { - //iDynTree::ModelExporter mdl_exporter; pfcSession_ptr session_ptr = pfcGetProESession(); pfcModel_ptr model_ptr = session_ptr->GetCurrentModel(); pfcSolid_ptr solid_ptr = pfcSolid::cast(session_ptr->GetCurrentModel()); - // TODO Principal units probably to be changed from MM to M before getting the model properties //auto length_unit = solid_ptr->GetPrincipalUnits()->GetUnit(pfcUnitType::pfcUNIT_LENGTH); // length_unit->Modify(pfcUnitConversionFactor::Create(0.001), length_unit->GetReferenceUnit()); // IT DOES NOT WORK - + // Export stl of the model iDynTree::Model idyn_model; //mdl_exporter.init(idyn_model); - + auto asm_component_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_FEATURE); if (asm_component_list->getarraysize() == 0) { - printToMessageWindow(session_ptr, "There are no CYS in the asm"); + printToMessageWindow(session_ptr, "There are no FEATURES in the asm", c2uLogLevel::WARN); return; } @@ -132,95 +138,92 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { auto feat = pfcFeature::cast(comp); auto feat_id = feat->GetId(); - if (feat->GetFeatType() == pfcFeatureType::pfcFEATTYPE_COMPONENT) + if (feat->GetFeatType() != pfcFeatureType::pfcFEATTYPE_COMPONENT) { - - auto modelhdl = session_ptr->RetrieveModel(pfcComponentFeat::cast(feat)->GetModelDescr()); - xintsequence_ptr seq = xintsequence::create(); - - seq->append(feat_id); - - pfcComponentPath_ptr comp_path = pfcCreateComponentPath(pfcAssembly::cast(model_ptr), seq); - - auto transform = comp_path->GetTransform(xfalse); - - auto m = transform->GetMatrix(); - auto o = transform->GetOrigin(); - printToMessageWindow(session_ptr, "feat name: id: " + to_string(feat_id)); - - printToMessageWindow(session_ptr, "origin x: " + to_string(o->get(0)) + " y: " + to_string(o->get(1)) + " z: " + to_string(o->get(2))); - printToMessageWindow(session_ptr, "transform:"); - printToMessageWindow(session_ptr, to_string(m->get(0, 0)) + " " + to_string(m->get(0, 1)) + " " + to_string(m->get(0, 2))); - printToMessageWindow(session_ptr, to_string(m->get(1, 0)) + " " + to_string(m->get(1, 1)) + " " + to_string(m->get(1, 2))); - printToMessageWindow(session_ptr, to_string(m->get(2, 0)) + " " + to_string(m->get(2, 1)) + " " + to_string(m->get(2, 2))); - - auto name = modelhdl->GetFullName(); - auto mass_prop = pfcSolid::cast(modelhdl)->GetMassProperty(); - auto com = mass_prop->GetGravityCenter(); - auto comInertia = mass_prop->GetCenterGravityInertiaTensor(); // TODO GetCoordSysInertia ? - - printToMessageWindow(session_ptr, "Model name is " + std::string(name) + " and weighs " + to_string(mass_prop->GetMass())); - printToMessageWindow(session_ptr, "Center of mass: x: " + to_string(com->get(0)) + " y: " + to_string(com->get(1)) + " z: " + to_string(com->get(2))); - printToMessageWindow(session_ptr, "Inertia tensor:"); - printToMessageWindow(session_ptr, to_string(comInertia->get(0, 0)) + " " + to_string(comInertia->get(0, 1)) + " " + to_string(comInertia->get(0, 2))); - printToMessageWindow(session_ptr, to_string(comInertia->get(1, 0)) + " " + to_string(comInertia->get(1, 1)) + " " + to_string(comInertia->get(1, 2))); - printToMessageWindow(session_ptr, to_string(comInertia->get(2, 0)) + " " + to_string(comInertia->get(2, 1)) + " " + to_string(comInertia->get(2, 2))); - - - auto csys_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); - if (csys_list->getarraysize() == 0) { - printToMessageWindow(session_ptr, "There are no CYS in the part " + string(name)); - return; - } + continue; + } - auto axes_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_AXIS); - printToMessageWindow(session_ptr, "There are " + to_string(axes_list->getarraysize()) + " axes"); - if (axes_list->getarraysize() == 0) { - printToMessageWindow(session_ptr, "There are no AXIS in the part " + string(name)); - return; - } + auto modelhdl = session_ptr->RetrieveModel(pfcComponentFeat::cast(feat)->GetModelDescr()); + xintsequence_ptr seq = xintsequence::create(); - // TODO We assume we have 1 axis and it is the one of the joint - auto axis = pfcAxis::cast(axes_list->get(0)); - printToMessageWindow(session_ptr, "The axis is called " + string(axis->GetName()) + " axes"); + seq->append(feat_id); - auto axis_data = wfcWAxis::cast(axis)->GetAxisData(); + pfcComponentPath_ptr comp_path = pfcCreateComponentPath(pfcAssembly::cast(model_ptr), seq); - auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell + auto transform = comp_path->GetTransform(xfalse); - auto unit = computeUnitVectorFromAxis(axis_line); + auto m = transform->GetMatrix(); + auto o = transform->GetOrigin(); + printToMessageWindow(session_ptr, "feat name: id: " + to_string(feat_id)); - printToMessageWindow(session_ptr, "unit vector of axis " + string(axis->GetName()) + " is : (" + std::to_string(unit[0]) + ", " + std::to_string(unit[1]) + ", " + std::to_string(unit[2]) + ")"); + printToMessageWindow(session_ptr, "origin x: " + to_string(o->get(0)) + " y: " + to_string(o->get(1)) + " z: " + to_string(o->get(2))); + printToMessageWindow(session_ptr, "transform:"); + printToMessageWindow(session_ptr, to_string(m->get(0, 0)) + " " + to_string(m->get(0, 1)) + " " + to_string(m->get(0, 2))); + printToMessageWindow(session_ptr, to_string(m->get(1, 0)) + " " + to_string(m->get(1, 1)) + " " + to_string(m->get(1, 2))); + printToMessageWindow(session_ptr, to_string(m->get(2, 0)) + " " + to_string(m->get(2, 1)) + " " + to_string(m->get(2, 2))); - // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it - modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create(csys_list->get(0)->GetName()))); + auto name = modelhdl->GetFullName(); + auto mass_prop = pfcSolid::cast(modelhdl)->GetMassProperty(); + auto com = mass_prop->GetGravityCenter(); + auto comInertia = mass_prop->GetCenterGravityInertiaTensor(); // TODO GetCoordSysInertia ? + printToMessageWindow(session_ptr, "Model name is " + std::string(name) + " and weighs " + to_string(mass_prop->GetMass())); + printToMessageWindow(session_ptr, "Center of mass: x: " + to_string(com->get(0)) + " y: " + to_string(com->get(1)) + " z: " + to_string(com->get(2))); + printToMessageWindow(session_ptr, "Inertia tensor:"); + printToMessageWindow(session_ptr, to_string(comInertia->get(0, 0)) + " " + to_string(comInertia->get(0, 1)) + " " + to_string(comInertia->get(0, 2))); + printToMessageWindow(session_ptr, to_string(comInertia->get(1, 0)) + " " + to_string(comInertia->get(1, 1)) + " " + to_string(comInertia->get(1, 2))); + printToMessageWindow(session_ptr, to_string(comInertia->get(2, 0)) + " " + to_string(comInertia->get(2, 1)) + " " + to_string(comInertia->get(2, 2))); - link.setInertia(fromCreo(mass_prop)); - if (i == asm_component_list->getarraysize()-1) { // TODO This is valid only for twobars - printToMessageWindow(session_ptr, "I AM ADDING A JOINT!"); - iDynTree::RevoluteJoint joint(fromCreo(transform), { {unit[0], unit[1], unit[2]}, - { o->get(0), o->get(1), o->get(2)} }); - - if (idyn_model.addJointAndLink(prevLinkName, prevLinkName + "--" + string(name), &joint, string(name), link) == iDynTree::JOINT_INVALID_INDEX) { - printToMessageWindow(session_ptr, "FAILED TO ADD JOINT!"); - return; - } - } - else { - prevLinkName = string(name); - idyn_model.addLink(string(name), link); + auto csys_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); + if (csys_list->getarraysize() == 0) { + printToMessageWindow(session_ptr, "There are no CYS in the part " + string(name)); + return; + } - } - //idyn_model.addAdditionalFrameToLink(string(name), string(name) + "_" + string(csys_list->get(0)->GetName()), fromCreo(transform)); TODO when we have an additional frame to add + auto axes_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_AXIS); + printToMessageWindow(session_ptr, "There are " + to_string(axes_list->getarraysize()) + " axes"); + if (axes_list->getarraysize() == 0) { + printToMessageWindow(session_ptr, "There are no AXIS in the part " + string(name)); + return; + } + + // TODO We assume we have 1 axis and it is the one of the joint + auto axis = pfcAxis::cast(axes_list->get(0)); + printToMessageWindow(session_ptr, "The axis is called " + string(axis->GetName()) + " axes"); + + auto axis_data = wfcWAxis::cast(axis)->GetAxisData(); + + auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell + + auto unit = computeUnitVectorFromAxis(axis_line); + + printToMessageWindow(session_ptr, "unit vector of axis " + string(axis->GetName()) + " is : (" + std::to_string(unit[0]) + ", " + std::to_string(unit[1]) + ", " + std::to_string(unit[2]) + ")"); - links_map[feat_id] = link; + // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it + modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create(csys_list->get(0)->GetName()))); + + link.setInertia(fromCreo(mass_prop)); + if (i == asm_component_list->getarraysize() - 1) { // TODO This is valid only for twobars + iDynTree::RevoluteJoint joint(fromCreo(transform), { {unit[0], unit[1], unit[2]}, + { o->get(0), o->get(1), o->get(2)} }); + + if (idyn_model.addJointAndLink(prevLinkName, prevLinkName + "--" + string(name), &joint, string(name), link) == iDynTree::JOINT_INVALID_INDEX) { + printToMessageWindow(session_ptr, "FAILED TO ADD JOINT!"); + return; + } + } + else + { + prevLinkName = string(name); + idyn_model.addLink(string(name), link); } - } - printToMessageWindow(session_ptr, "idynModel " + idyn_model.toString()); - + //idyn_model.addAdditionalFrameToLink(string(name), string(name) + "_" + string(csys_list->get(0)->GetName()), fromCreo(transform)); TODO when we have an additional frame to add + links_map[feat_id] = link; + } + + printToMessageWindow(session_ptr, "idynModel " + idyn_model.toString()); //if (!mdl_exporter.exportModelToFile("model.urdf")) { // printToMessageWindow(session_ptr, "Error exporting the urdf"); @@ -230,7 +233,6 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { } }; - class Creo2UrdfAccessListener : public pfcUICommandAccessListener { public: pfcCommandAccess OnCommandAccess(xbool AllowErrorMessages) override { @@ -243,7 +245,6 @@ class Creo2UrdfAccessListener : public pfcUICommandAccessListener { return pfcCommandAccess::pfcACCESS_UNAVAILABLE; } return pfcCommandAccess::pfcACCESS_AVAILABLE; - } }; @@ -255,9 +256,9 @@ PURPOSE : \*====================================================================*/ extern "C" int user_initialize( int argc, - char *argv[], - char *version, - char *build, + char* argv[], + char* version, + char* build, wchar_t errbuf[80]) { auto session = pfcGetProESession(); @@ -268,7 +269,6 @@ extern "C" int user_initialize( //session->RibbonDefinitionfileLoad("tool.rbn"); - return (0); } @@ -278,5 +278,4 @@ PURPOSE : To handle any termination actions \*====================================================================*/ extern "C" void user_terminate() { - -} +} \ No newline at end of file diff --git a/src/creo2urdf/text/usascii/creo2urdf.txt b/src/creo2urdf/text/usascii/creo2urdf.txt index 083c925..0e7090f 100644 --- a/src/creo2urdf/text/usascii/creo2urdf.txt +++ b/src/creo2urdf/text/usascii/creo2urdf.txt @@ -1,13 +1,9 @@ --Run Creo2Urdf --Run Creo2Urdf +%CIc2uINFO +%0s # # -Run Creo2Urdf code -Run Creo2Urdf code -# -# -%CW"DEBUG %0s" -"%0s" + +%CWc2uWARN +%0s # # - From e89b708744bb9882cfe582509e93c89d8fc47ded Mon Sep 17 00:00:00 2001 From: Nicogene Date: Tue, 11 Apr 2023 10:24:57 +0200 Subject: [PATCH 15/21] creo2urdf: fix tabs with spaces --- src/creo2urdf/src/Creo2Urdf.cpp | 327 ++++++++++++++++---------------- 1 file changed, 164 insertions(+), 163 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index e6f9dab..ec243b1 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -30,21 +30,21 @@ constexpr double epsilon = 1e-12; enum class c2uLogLevel { - INFO = 0, - WARN + INFO = 0, + WARN }; std::map log_level_key = { - {c2uLogLevel::INFO, "c2uINFO"}, - {c2uLogLevel::WARN, "c2uWARN"} + {c2uLogLevel::INFO, "c2uINFO"}, + {c2uLogLevel::WARN, "c2uWARN"} }; void printToMessageWindow(pfcSession_ptr session, std::string message, c2uLogLevel log_level = c2uLogLevel::INFO) { - xstringsequence_ptr msg_sequence = xstringsequence::create(); - msg_sequence->append(xstring(message.c_str())); - session->UIClearMessage(); - session->UIDisplayMessage("creo2urdf.txt", log_level_key[log_level].c_str(), msg_sequence); + xstringsequence_ptr msg_sequence = xstringsequence::create(); + msg_sequence->append(xstring(message.c_str())); + session->UIClearMessage(); + session->UIDisplayMessage("creo2urdf.txt", log_level_key[log_level].c_str(), msg_sequence); } @@ -55,197 +55,198 @@ std::map, iDynTree::IJointPtr> joints_map; std::array computeUnitVectorFromAxis(pfcCurveDescriptor_ptr axis_data) { - auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell + auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell - // There are just two points in the array - pfcPoint3D_ptr pstart = axis_line->GetEnd1(); - pfcPoint3D_ptr pend = axis_line->GetEnd2(); + // There are just two points in the array + pfcPoint3D_ptr pstart = axis_line->GetEnd1(); + pfcPoint3D_ptr pend = axis_line->GetEnd2(); - std::array unit_vector; + std::array unit_vector; - double module = sqrt(pow(pend->get(0) - pstart->get(0), 2) + - pow(pend->get(1) - pstart->get(1), 2) + - pow(pend->get(2) - pstart->get(2), 2)); + double module = sqrt(pow(pend->get(0) - pstart->get(0), 2) + + pow(pend->get(1) - pstart->get(1), 2) + + pow(pend->get(2) - pstart->get(2), 2)); - if (module < epsilon) - { - return unit_vector; - } + if (module < epsilon) + { + return unit_vector; + } - unit_vector[0] = (pend->get(0) - pstart->get(0)) / module; - unit_vector[1] = (pend->get(1) - pstart->get(1)) / module; - unit_vector[2] = (pend->get(2) - pstart->get(2)) / module; + unit_vector[0] = (pend->get(0) - pstart->get(0)) / module; + unit_vector[1] = (pend->get(1) - pstart->get(1)) / module; + unit_vector[2] = (pend->get(2) - pstart->get(2)) / module; - return unit_vector; + return unit_vector; } iDynTree::SpatialInertia fromCreo(pfcMassProperty_ptr mass_prop) { - auto com = mass_prop->GetGravityCenter(); - auto inertia_tensor = mass_prop->GetCenterGravityInertiaTensor(); - iDynTree::RotationalInertiaRaw idyn_inertia_tensor = iDynTree::RotationalInertiaRaw::Zero(); - for (int i_row = 0; i_row < idyn_inertia_tensor.rows(); i_row++) { - for (int j_col = 0; j_col < idyn_inertia_tensor.cols(); j_col++) { - idyn_inertia_tensor.setVal(i_row, j_col, inertia_tensor->get(i_row, j_col)); - } - } - - iDynTree::SpatialInertia sp_inertia(mass_prop->GetMass(), - { com->get(0), com->get(1), com->get(2) }, - idyn_inertia_tensor); - return sp_inertia; + auto com = mass_prop->GetGravityCenter(); + auto inertia_tensor = mass_prop->GetCenterGravityInertiaTensor(); + iDynTree::RotationalInertiaRaw idyn_inertia_tensor = iDynTree::RotationalInertiaRaw::Zero(); + for (int i_row = 0; i_row < idyn_inertia_tensor.rows(); i_row++) { + for (int j_col = 0; j_col < idyn_inertia_tensor.cols(); j_col++) { + idyn_inertia_tensor.setVal(i_row, j_col, inertia_tensor->get(i_row, j_col)); + } + } + + iDynTree::SpatialInertia sp_inertia(mass_prop->GetMass(), + { com->get(0), com->get(1), com->get(2) }, + idyn_inertia_tensor); + return sp_inertia; } iDynTree::Transform fromCreo(pfcTransform3D_ptr creo_trf) { - iDynTree::Transform idyn_trf; - auto o = creo_trf->GetOrigin(); - auto m = creo_trf->GetMatrix(); - idyn_trf.setPosition({ o->get(0), o->get(1), o->get(2) }); - idyn_trf.setRotation({ m->get(0,0), m->get(0,1), m->get(0,2), - m->get(1,0), m->get(1,1), m->get(1,2), - m->get(2,0), m->get(2,1), m->get(2,2) }); - return idyn_trf; + iDynTree::Transform idyn_trf; + auto o = creo_trf->GetOrigin(); + auto m = creo_trf->GetMatrix(); + idyn_trf.setPosition({ o->get(0), o->get(1), o->get(2) }); + idyn_trf.setRotation({ m->get(0,0), m->get(0,1), m->get(0,2), + m->get(1,0), m->get(1,1), m->get(1,2), + m->get(2,0), m->get(2,1), m->get(2,2) }); + + return idyn_trf; } class Creo2UrdfActionListerner : public pfcUICommandActionListener { public: - void OnCommand() override { - //iDynTree::ModelExporter mdl_exporter; - pfcSession_ptr session_ptr = pfcGetProESession(); - pfcModel_ptr model_ptr = session_ptr->GetCurrentModel(); - pfcSolid_ptr solid_ptr = pfcSolid::cast(session_ptr->GetCurrentModel()); - - // TODO Principal units probably to be changed from MM to M before getting the model properties - //auto length_unit = solid_ptr->GetPrincipalUnits()->GetUnit(pfcUnitType::pfcUNIT_LENGTH); - // length_unit->Modify(pfcUnitConversionFactor::Create(0.001), length_unit->GetReferenceUnit()); // IT DOES NOT WORK - - // Export stl of the model - - iDynTree::Model idyn_model; - //mdl_exporter.init(idyn_model); + void OnCommand() override { + //iDynTree::ModelExporter mdl_exporter; + pfcSession_ptr session_ptr = pfcGetProESession(); + pfcModel_ptr model_ptr = session_ptr->GetCurrentModel(); + pfcSolid_ptr solid_ptr = pfcSolid::cast(session_ptr->GetCurrentModel()); + + // TODO Principal units probably to be changed from MM to M before getting the model properties + //auto length_unit = solid_ptr->GetPrincipalUnits()->GetUnit(pfcUnitType::pfcUNIT_LENGTH); + // length_unit->Modify(pfcUnitConversionFactor::Create(0.001), length_unit->GetReferenceUnit()); // IT DOES NOT WORK + + // Export stl of the model + + iDynTree::Model idyn_model; + //mdl_exporter.init(idyn_model); - auto asm_component_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_FEATURE); - if (asm_component_list->getarraysize() == 0) { - printToMessageWindow(session_ptr, "There are no FEATURES in the asm", c2uLogLevel::WARN); - return; - } + auto asm_component_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_FEATURE); + if (asm_component_list->getarraysize() == 0) { + printToMessageWindow(session_ptr, "There are no FEATURES in the asm", c2uLogLevel::WARN); + return; + } - std::string prevLinkName = ""; + std::string prevLinkName = ""; - for (int i = 0; i < asm_component_list->getarraysize(); i++) - { - iDynTree::Link link; - auto comp = asm_component_list->get(i); - auto feat = pfcFeature::cast(comp); - auto feat_id = feat->GetId(); + for (int i = 0; i < asm_component_list->getarraysize(); i++) + { + iDynTree::Link link; + auto comp = asm_component_list->get(i); + auto feat = pfcFeature::cast(comp); + auto feat_id = feat->GetId(); - if (feat->GetFeatType() != pfcFeatureType::pfcFEATTYPE_COMPONENT) - { - continue; - } + if (feat->GetFeatType() != pfcFeatureType::pfcFEATTYPE_COMPONENT) + { + continue; + } - auto modelhdl = session_ptr->RetrieveModel(pfcComponentFeat::cast(feat)->GetModelDescr()); - xintsequence_ptr seq = xintsequence::create(); + auto modelhdl = session_ptr->RetrieveModel(pfcComponentFeat::cast(feat)->GetModelDescr()); + xintsequence_ptr seq = xintsequence::create(); - seq->append(feat_id); + seq->append(feat_id); - pfcComponentPath_ptr comp_path = pfcCreateComponentPath(pfcAssembly::cast(model_ptr), seq); + pfcComponentPath_ptr comp_path = pfcCreateComponentPath(pfcAssembly::cast(model_ptr), seq); - auto transform = comp_path->GetTransform(xfalse); + auto transform = comp_path->GetTransform(xfalse); - auto m = transform->GetMatrix(); - auto o = transform->GetOrigin(); - printToMessageWindow(session_ptr, "feat name: id: " + to_string(feat_id)); + auto m = transform->GetMatrix(); + auto o = transform->GetOrigin(); + printToMessageWindow(session_ptr, "feat name: id: " + to_string(feat_id)); - printToMessageWindow(session_ptr, "origin x: " + to_string(o->get(0)) + " y: " + to_string(o->get(1)) + " z: " + to_string(o->get(2))); - printToMessageWindow(session_ptr, "transform:"); - printToMessageWindow(session_ptr, to_string(m->get(0, 0)) + " " + to_string(m->get(0, 1)) + " " + to_string(m->get(0, 2))); - printToMessageWindow(session_ptr, to_string(m->get(1, 0)) + " " + to_string(m->get(1, 1)) + " " + to_string(m->get(1, 2))); - printToMessageWindow(session_ptr, to_string(m->get(2, 0)) + " " + to_string(m->get(2, 1)) + " " + to_string(m->get(2, 2))); + printToMessageWindow(session_ptr, "origin x: " + to_string(o->get(0)) + " y: " + to_string(o->get(1)) + " z: " + to_string(o->get(2))); + printToMessageWindow(session_ptr, "transform:"); + printToMessageWindow(session_ptr, to_string(m->get(0, 0)) + " " + to_string(m->get(0, 1)) + " " + to_string(m->get(0, 2))); + printToMessageWindow(session_ptr, to_string(m->get(1, 0)) + " " + to_string(m->get(1, 1)) + " " + to_string(m->get(1, 2))); + printToMessageWindow(session_ptr, to_string(m->get(2, 0)) + " " + to_string(m->get(2, 1)) + " " + to_string(m->get(2, 2))); - auto name = modelhdl->GetFullName(); - auto mass_prop = pfcSolid::cast(modelhdl)->GetMassProperty(); - auto com = mass_prop->GetGravityCenter(); - auto comInertia = mass_prop->GetCenterGravityInertiaTensor(); // TODO GetCoordSysInertia ? + auto name = modelhdl->GetFullName(); + auto mass_prop = pfcSolid::cast(modelhdl)->GetMassProperty(); + auto com = mass_prop->GetGravityCenter(); + auto comInertia = mass_prop->GetCenterGravityInertiaTensor(); // TODO GetCoordSysInertia ? - printToMessageWindow(session_ptr, "Model name is " + std::string(name) + " and weighs " + to_string(mass_prop->GetMass())); - printToMessageWindow(session_ptr, "Center of mass: x: " + to_string(com->get(0)) + " y: " + to_string(com->get(1)) + " z: " + to_string(com->get(2))); - printToMessageWindow(session_ptr, "Inertia tensor:"); - printToMessageWindow(session_ptr, to_string(comInertia->get(0, 0)) + " " + to_string(comInertia->get(0, 1)) + " " + to_string(comInertia->get(0, 2))); - printToMessageWindow(session_ptr, to_string(comInertia->get(1, 0)) + " " + to_string(comInertia->get(1, 1)) + " " + to_string(comInertia->get(1, 2))); - printToMessageWindow(session_ptr, to_string(comInertia->get(2, 0)) + " " + to_string(comInertia->get(2, 1)) + " " + to_string(comInertia->get(2, 2))); + printToMessageWindow(session_ptr, "Model name is " + std::string(name) + " and weighs " + to_string(mass_prop->GetMass())); + printToMessageWindow(session_ptr, "Center of mass: x: " + to_string(com->get(0)) + " y: " + to_string(com->get(1)) + " z: " + to_string(com->get(2))); + printToMessageWindow(session_ptr, "Inertia tensor:"); + printToMessageWindow(session_ptr, to_string(comInertia->get(0, 0)) + " " + to_string(comInertia->get(0, 1)) + " " + to_string(comInertia->get(0, 2))); + printToMessageWindow(session_ptr, to_string(comInertia->get(1, 0)) + " " + to_string(comInertia->get(1, 1)) + " " + to_string(comInertia->get(1, 2))); + printToMessageWindow(session_ptr, to_string(comInertia->get(2, 0)) + " " + to_string(comInertia->get(2, 1)) + " " + to_string(comInertia->get(2, 2))); - auto csys_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); - if (csys_list->getarraysize() == 0) { - printToMessageWindow(session_ptr, "There are no CYS in the part " + string(name)); - return; - } + auto csys_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_COORD_SYS); + if (csys_list->getarraysize() == 0) { + printToMessageWindow(session_ptr, "There are no CYS in the part " + string(name)); + return; + } - auto axes_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_AXIS); - printToMessageWindow(session_ptr, "There are " + to_string(axes_list->getarraysize()) + " axes"); - if (axes_list->getarraysize() == 0) { - printToMessageWindow(session_ptr, "There are no AXIS in the part " + string(name)); - return; - } + auto axes_list = modelhdl->ListItems(pfcModelItemType::pfcITEM_AXIS); + printToMessageWindow(session_ptr, "There are " + to_string(axes_list->getarraysize()) + " axes"); + if (axes_list->getarraysize() == 0) { + printToMessageWindow(session_ptr, "There are no AXIS in the part " + string(name)); + return; + } - // TODO We assume we have 1 axis and it is the one of the joint - auto axis = pfcAxis::cast(axes_list->get(0)); - printToMessageWindow(session_ptr, "The axis is called " + string(axis->GetName()) + " axes"); + // TODO We assume we have 1 axis and it is the one of the joint + auto axis = pfcAxis::cast(axes_list->get(0)); + printToMessageWindow(session_ptr, "The axis is called " + string(axis->GetName()) + " axes"); - auto axis_data = wfcWAxis::cast(axis)->GetAxisData(); + auto axis_data = wfcWAxis::cast(axis)->GetAxisData(); - auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell + auto axis_line = pfcLineDescriptor::cast(axis_data); // cursed cast from hell - auto unit = computeUnitVectorFromAxis(axis_line); + auto unit = computeUnitVectorFromAxis(axis_line); - printToMessageWindow(session_ptr, "unit vector of axis " + string(axis->GetName()) + " is : (" + std::to_string(unit[0]) + ", " + std::to_string(unit[1]) + ", " + std::to_string(unit[2]) + ")"); + printToMessageWindow(session_ptr, "unit vector of axis " + string(axis->GetName()) + " is : (" + std::to_string(unit[0]) + ", " + std::to_string(unit[1]) + ", " + std::to_string(unit[2]) + ")"); - // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it - modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create(csys_list->get(0)->GetName()))); + // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it + modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create(csys_list->get(0)->GetName()))); - link.setInertia(fromCreo(mass_prop)); - if (i == asm_component_list->getarraysize() - 1) { // TODO This is valid only for twobars - iDynTree::RevoluteJoint joint(fromCreo(transform), { {unit[0], unit[1], unit[2]}, - { o->get(0), o->get(1), o->get(2)} }); + link.setInertia(fromCreo(mass_prop)); + if (i == asm_component_list->getarraysize() - 1) { // TODO This is valid only for twobars + iDynTree::RevoluteJoint joint(fromCreo(transform), { {unit[0], unit[1], unit[2]}, + { o->get(0), o->get(1), o->get(2)} }); - if (idyn_model.addJointAndLink(prevLinkName, prevLinkName + "--" + string(name), &joint, string(name), link) == iDynTree::JOINT_INVALID_INDEX) { - printToMessageWindow(session_ptr, "FAILED TO ADD JOINT!"); - return; - } - } - else - { - prevLinkName = string(name); - idyn_model.addLink(string(name), link); - } + if (idyn_model.addJointAndLink(prevLinkName, prevLinkName + "--" + string(name), &joint, string(name), link) == iDynTree::JOINT_INVALID_INDEX) { + printToMessageWindow(session_ptr, "FAILED TO ADD JOINT!"); + return; + } + } + else + { + prevLinkName = string(name); + idyn_model.addLink(string(name), link); + } - //idyn_model.addAdditionalFrameToLink(string(name), string(name) + "_" + string(csys_list->get(0)->GetName()), fromCreo(transform)); TODO when we have an additional frame to add + //idyn_model.addAdditionalFrameToLink(string(name), string(name) + "_" + string(csys_list->get(0)->GetName()), fromCreo(transform)); TODO when we have an additional frame to add - links_map[feat_id] = link; - } + links_map[feat_id] = link; + } - printToMessageWindow(session_ptr, "idynModel " + idyn_model.toString()); + printToMessageWindow(session_ptr, "idynModel " + idyn_model.toString()); - //if (!mdl_exporter.exportModelToFile("model.urdf")) { - // printToMessageWindow(session_ptr, "Error exporting the urdf"); - //} + //if (!mdl_exporter.exportModelToFile("model.urdf")) { + // printToMessageWindow(session_ptr, "Error exporting the urdf"); + //} - return; - } + return; + } }; class Creo2UrdfAccessListener : public pfcUICommandAccessListener { public: - pfcCommandAccess OnCommandAccess(xbool AllowErrorMessages) override { - auto model = pfcGetProESession()->GetCurrentModel(); - if (!model) { - return pfcCommandAccess::pfcACCESS_AVAILABLE; - } - auto type = model->GetType(); - if (type != pfcMDL_PART && type != pfcMDL_ASSEMBLY) { - return pfcCommandAccess::pfcACCESS_UNAVAILABLE; - } - return pfcCommandAccess::pfcACCESS_AVAILABLE; - } + pfcCommandAccess OnCommandAccess(xbool AllowErrorMessages) override { + auto model = pfcGetProESession()->GetCurrentModel(); + if (!model) { + return pfcCommandAccess::pfcACCESS_AVAILABLE; + } + auto type = model->GetType(); + if (type != pfcMDL_PART && type != pfcMDL_ASSEMBLY) { + return pfcCommandAccess::pfcACCESS_UNAVAILABLE; + } + return pfcCommandAccess::pfcACCESS_AVAILABLE; + } }; static ProError status; @@ -255,21 +256,21 @@ FUNCTION : user_initialize() PURPOSE : \*====================================================================*/ extern "C" int user_initialize( - int argc, - char* argv[], - char* version, - char* build, - wchar_t errbuf[80]) + int argc, + char* argv[], + char* version, + char* build, + wchar_t errbuf[80]) { - auto session = pfcGetProESession(); + auto session = pfcGetProESession(); - auto cmd = session->UICreateCommand("Creo2Urdf", new Creo2UrdfActionListerner()); - cmd->AddActionListener(new Creo2UrdfAccessListener()); // To be checked it is odd - cmd->Designate("ui.txt", "Run Creo2Urdf", "Run Creo2Urdf", "Run Creo2Urdf"); + auto cmd = session->UICreateCommand("Creo2Urdf", new Creo2UrdfActionListerner()); + cmd->AddActionListener(new Creo2UrdfAccessListener()); // To be checked it is odd + cmd->Designate("ui.txt", "Run Creo2Urdf", "Run Creo2Urdf", "Run Creo2Urdf"); - //session->RibbonDefinitionfileLoad("tool.rbn"); + //session->RibbonDefinitionfileLoad("tool.rbn"); - return (0); + return (0); } /*====================================================================*\ From 65744f071cb2c754821fa5b8da15fd9cf72abd2a Mon Sep 17 00:00:00 2001 From: Nicogene Date: Wed, 12 Apr 2023 15:06:55 +0200 Subject: [PATCH 16/21] creo2urdf: model export fail --- CMakeLists.txt | 1 + src/creo2urdf/CMakeLists.txt | 10 +++-- src/creo2urdf/include/creo2urdf/Creo2Urdf.h | 7 +++ src/creo2urdf/src/Creo2Urdf.cpp | 49 ++++++++++++++------- src/creo2urdf/text/usascii/messages.txt | 32 -------------- 5 files changed, 46 insertions(+), 53 deletions(-) delete mode 100644 src/creo2urdf/text/usascii/messages.txt diff --git a/CMakeLists.txt b/CMakeLists.txt index 06b9bfe..4e51435 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -12,6 +12,7 @@ include(GNUInstallDirs) include(FeatureSummary) find_package(YCM 0.12 REQUIRED) +find_package(Eigen3 REQUIRED) find_package(iDynTree REQUIRED) list(APPEND CMAKE_MODULE_PATH ${CMAKE_CURRENT_SOURCE_DIR}/cmake) diff --git a/src/creo2urdf/CMakeLists.txt b/src/creo2urdf/CMakeLists.txt index cf09759..7437e9f 100644 --- a/src/creo2urdf/CMakeLists.txt +++ b/src/creo2urdf/CMakeLists.txt @@ -56,7 +56,9 @@ add_compile_definitions(creo2urdf PUBLIC PRO_MACHINE=36 PRO_OS=4) # Link dependencies -target_link_libraries(creo2urdf PUBLIC protk_dllmd_NU +target_link_libraries(creo2urdf PRIVATE iDynTree::idyntree-modelio + iDynTree::idyntree-high-level + protk_dllmd_NU otk_cpp_md otk_no222_md ucore @@ -65,9 +67,9 @@ target_link_libraries(creo2urdf PUBLIC protk_dllmd_NU ws2_32 advapi32 mpr - netapi32 - iDynTree::idyntree-modelio - iDynTree::idyntree-high-level) + netapi32) + + # FIXME all these win32 libraries that are used by protk dll have to be set as dependencies of the target ## FIXME diff --git a/src/creo2urdf/include/creo2urdf/Creo2Urdf.h b/src/creo2urdf/include/creo2urdf/Creo2Urdf.h index e69de29..7114896 100644 --- a/src/creo2urdf/include/creo2urdf/Creo2Urdf.h +++ b/src/creo2urdf/include/creo2urdf/Creo2Urdf.h @@ -0,0 +1,7 @@ +/* + * Copyright (C) 2006-2023 Istituto Italiano di Tecnologia (IIT) + * All rights reserved. + * + * This software may be modified and distributed under the terms of the + * BSD-3-Clause license. See the accompanying LICENSE file for details. + */ \ No newline at end of file diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index ec243b1..ea0fa74 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -1,12 +1,10 @@ -#include -#include -#include -#include -#include -#include -#include -#include -#include +/* + * Copyright (C) 2006-2023 Istituto Italiano di Tecnologia (IIT) + * All rights reserved. + * + * This software may be modified and distributed under the terms of the + * BSD-3-Clause license. See the accompanying LICENSE file for details. + */ #include #include @@ -22,12 +20,18 @@ #include #include #include +#include +#include + + #include #include #include + constexpr double epsilon = 1e-12; + enum class c2uLogLevel { INFO = 0, @@ -110,7 +114,9 @@ iDynTree::Transform fromCreo(pfcTransform3D_ptr creo_trf) { class Creo2UrdfActionListerner : public pfcUICommandActionListener { public: void OnCommand() override { - //iDynTree::ModelExporter mdl_exporter; + std::ofstream file; + file.open("model_manual.urdf"); + iDynTree::ModelExporter mdl_exporter; pfcSession_ptr session_ptr = pfcGetProESession(); pfcModel_ptr model_ptr = session_ptr->GetCurrentModel(); pfcSolid_ptr solid_ptr = pfcSolid::cast(session_ptr->GetCurrentModel()); @@ -119,10 +125,11 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { //auto length_unit = solid_ptr->GetPrincipalUnits()->GetUnit(pfcUnitType::pfcUNIT_LENGTH); // length_unit->Modify(pfcUnitConversionFactor::Create(0.001), length_unit->GetReferenceUnit()); // IT DOES NOT WORK - // Export stl of the model - iDynTree::Model idyn_model; - //mdl_exporter.init(idyn_model); + mdl_exporter.init(idyn_model); + //iDynTree::ModelExporterOptions export_options; + //export_options.robotExportedName = "2BARS"; + //mdl_exporter.setExportingOptions(export_options); auto asm_component_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_FEATURE); if (asm_component_list->getarraysize() == 0) { @@ -225,10 +232,19 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { } printToMessageWindow(session_ptr, "idynModel " + idyn_model.toString()); + std::string model_str = ""; + + if (!mdl_exporter.exportModelToString(model_str)) { + printToMessageWindow(session_ptr, "Error exporting the urdf"); + } + else { + file << model_str; + file.close(); + } - //if (!mdl_exporter.exportModelToFile("model.urdf")) { - // printToMessageWindow(session_ptr, "Error exporting the urdf"); - //} + if (!mdl_exporter.exportModelToFile("model.urdf")) { + printToMessageWindow(session_ptr, "Error exporting the urdf"); + } return; } @@ -249,7 +265,6 @@ class Creo2UrdfAccessListener : public pfcUICommandAccessListener { } }; -static ProError status; /*====================================================================*\ FUNCTION : user_initialize() diff --git a/src/creo2urdf/text/usascii/messages.txt b/src/creo2urdf/text/usascii/messages.txt deleted file mode 100644 index 822bca6..0000000 --- a/src/creo2urdf/text/usascii/messages.txt +++ /dev/null @@ -1,32 +0,0 @@ -Welcome -Hello World! -# -# -%CPMyPrompt -This is a prompt -# -# -%CIMyInfo -This is an info -# -# -%CEMyError -This is an error -# -# -%CWMyWarning -This is a warning -# -# -%CCMyCritical -This is a critical -# -# -DefaultMsg -%0s -# -# -DefaultMsg Wstr -%0w -# -# \ No newline at end of file From b3ad480b81ee1eb1db94ad8f1ff717443b93d601 Mon Sep 17 00:00:00 2001 From: Nicogene Date: Wed, 12 Apr 2023 15:32:45 +0200 Subject: [PATCH 17/21] creo2urdf: first urdf exportation working --- src/creo2urdf/src/Creo2Urdf.cpp | 21 +++++---------------- 1 file changed, 5 insertions(+), 16 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index ea0fa74..bd5e79a 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -20,8 +20,6 @@ #include #include #include -#include -#include #include @@ -114,8 +112,6 @@ iDynTree::Transform fromCreo(pfcTransform3D_ptr creo_trf) { class Creo2UrdfActionListerner : public pfcUICommandActionListener { public: void OnCommand() override { - std::ofstream file; - file.open("model_manual.urdf"); iDynTree::ModelExporter mdl_exporter; pfcSession_ptr session_ptr = pfcGetProESession(); pfcModel_ptr model_ptr = session_ptr->GetCurrentModel(); @@ -126,10 +122,8 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { // length_unit->Modify(pfcUnitConversionFactor::Create(0.001), length_unit->GetReferenceUnit()); // IT DOES NOT WORK iDynTree::Model idyn_model; - mdl_exporter.init(idyn_model); - //iDynTree::ModelExporterOptions export_options; - //export_options.robotExportedName = "2BARS"; - //mdl_exporter.setExportingOptions(export_options); + iDynTree::ModelExporterOptions export_options; + export_options.robotExportedName = "2BARS"; auto asm_component_list = model_ptr->ListItems(pfcModelItemType::pfcITEM_FEATURE); if (asm_component_list->getarraysize() == 0) { @@ -233,14 +227,9 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { printToMessageWindow(session_ptr, "idynModel " + idyn_model.toString()); std::string model_str = ""; - - if (!mdl_exporter.exportModelToString(model_str)) { - printToMessageWindow(session_ptr, "Error exporting the urdf"); - } - else { - file << model_str; - file.close(); - } + + mdl_exporter.init(idyn_model); + mdl_exporter.setExportingOptions(export_options); if (!mdl_exporter.exportModelToFile("model.urdf")) { printToMessageWindow(session_ptr, "Error exporting the urdf"); From 8df332cd2fdb7b007c06afa060648937668b9568 Mon Sep 17 00:00:00 2001 From: Nicogene Date: Wed, 12 Apr 2023 15:32:58 +0200 Subject: [PATCH 18/21] Add installation instructions --- README.md | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) diff --git a/README.md b/README.md index 1acd90d..36f53d1 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,23 @@ # creo2urdf Generate URDF model from CREO mechanisms. +## Installation + +### Dependencies +Right now `creo2urdf` plugin needs that the dependencies are compiled and linked **statically**: +- Download [vcpkg](https://github.com/microsoft/vcpkg): `git clone https://github.com/microsoft/vcpkg` +- Bootstrap vcpkg: `.\vcpkg\bootstrap-vcpkg.bat` +- Run `[path to vcpkg]/vcpkg install --triplet x64-windows-static-md eigen3 libxml2 assimp` +- Compile idyntree from source following the instructions in https://github.com/robotology/idyntree/blob/master/doc/build-from-source.md, remembering to pass `-DCMAKE_TOOLCHAIN_FILE=[path to vcpkg]/scripts/buildsystems/vcpkg.cmake -DVCPKG_TARGET_TRIPLET=x64-windows-static-md` and `-DBUILD_SHARED_LIBS:BOOL=OFF` to ensure that iDynTree is compiled as static. +- Install iDynTree + +### Build `creo2urdf` + +- Pass `-DCMAKE_TOOLCHAIN_FILE=[path to vcpkg]/scripts/buildsystems/vcpkg.cmake -DVCPKG_TARGET_TRIPLET=x64-windows-static-md` to the creo2urdf compilation, remembering also to pass the iDynTree's installaton location via `CMAKE_PREFIX_PATH`. + +For who uses CMake integrate in Visual Studio `-DCMAKE_TOOLCHAIN_FILE` option has not to be passed to `CMake command arguments` but in the `vcpkg.cmake` file path has to be set in `CMake toolchain file` + + ### Maintainers This repository is maintained by: From 4603298956a7c74c7b2849fd8602794735699ee8 Mon Sep 17 00:00:00 2001 From: Nicogene Date: Wed, 12 Apr 2023 16:03:26 +0200 Subject: [PATCH 19/21] Add idyntree issue --- CMakeLists.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/CMakeLists.txt b/CMakeLists.txt index 4e51435..1dc168e 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -12,6 +12,7 @@ include(GNUInstallDirs) include(FeatureSummary) find_package(YCM 0.12 REQUIRED) +# Needed for https://github.com/robotology/idyntree/issues/1065 find_package(Eigen3 REQUIRED) find_package(iDynTree REQUIRED) list(APPEND CMAKE_MODULE_PATH ${CMAKE_CURRENT_SOURCE_DIR}/cmake) From 910f5887c2d90fe3bf54fa40eaeea89cc91ca9ee Mon Sep 17 00:00:00 2001 From: Nicogene Date: Thu, 13 Apr 2023 10:28:15 +0200 Subject: [PATCH 20/21] creo2urdf: handle mm->m conversion --- src/creo2urdf/src/Creo2Urdf.cpp | 18 +++++++----------- 1 file changed, 7 insertions(+), 11 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index bd5e79a..48d580d 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -27,8 +27,9 @@ #include -constexpr double epsilon = 1e-12; - +constexpr double mm_to_m = 1e-3; +constexpr double mm2_to_m2 = 1e-6; +constexpr double epsilon = 1e-12; enum class c2uLogLevel { @@ -50,10 +51,6 @@ void printToMessageWindow(pfcSession_ptr session, std::string message, c2uLogLev } -// The key is id of the feature -std::map links_map; -// The key are id parent id child -std::map, iDynTree::IJointPtr> joints_map; std::array computeUnitVectorFromAxis(pfcCurveDescriptor_ptr axis_data) { @@ -87,12 +84,12 @@ iDynTree::SpatialInertia fromCreo(pfcMassProperty_ptr mass_prop) { iDynTree::RotationalInertiaRaw idyn_inertia_tensor = iDynTree::RotationalInertiaRaw::Zero(); for (int i_row = 0; i_row < idyn_inertia_tensor.rows(); i_row++) { for (int j_col = 0; j_col < idyn_inertia_tensor.cols(); j_col++) { - idyn_inertia_tensor.setVal(i_row, j_col, inertia_tensor->get(i_row, j_col)); + idyn_inertia_tensor.setVal(i_row, j_col, inertia_tensor->get(i_row, j_col) * mm2_to_m2); } } iDynTree::SpatialInertia sp_inertia(mass_prop->GetMass(), - { com->get(0), com->get(1), com->get(2) }, + { com->get(0) * mm_to_m, com->get(1) * mm_to_m, com->get(2) * mm_to_m }, idyn_inertia_tensor); return sp_inertia; } @@ -101,7 +98,7 @@ iDynTree::Transform fromCreo(pfcTransform3D_ptr creo_trf) { iDynTree::Transform idyn_trf; auto o = creo_trf->GetOrigin(); auto m = creo_trf->GetMatrix(); - idyn_trf.setPosition({ o->get(0), o->get(1), o->get(2) }); + idyn_trf.setPosition({ o->get(0) * mm_to_m, o->get(1) * mm_to_m, o->get(2) * mm_to_m }); idyn_trf.setRotation({ m->get(0,0), m->get(0,1), m->get(0,2), m->get(1,0), m->get(1,1), m->get(1,2), m->get(2,0), m->get(2,1), m->get(2,2) }); @@ -207,7 +204,7 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { link.setInertia(fromCreo(mass_prop)); if (i == asm_component_list->getarraysize() - 1) { // TODO This is valid only for twobars iDynTree::RevoluteJoint joint(fromCreo(transform), { {unit[0], unit[1], unit[2]}, - { o->get(0), o->get(1), o->get(2)} }); + { o->get(0) * mm_to_m, o->get(1) * mm_to_m, o->get(2) * mm_to_m } }); if (idyn_model.addJointAndLink(prevLinkName, prevLinkName + "--" + string(name), &joint, string(name), link) == iDynTree::JOINT_INVALID_INDEX) { printToMessageWindow(session_ptr, "FAILED TO ADD JOINT!"); @@ -222,7 +219,6 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { //idyn_model.addAdditionalFrameToLink(string(name), string(name) + "_" + string(csys_list->get(0)->GetName()), fromCreo(transform)); TODO when we have an additional frame to add - links_map[feat_id] = link; } printToMessageWindow(session_ptr, "idynModel " + idyn_model.toString()); From 115fe0f0920e232d9f8fc0d135066b58fd46d48f Mon Sep 17 00:00:00 2001 From: Nicogene Date: Thu, 13 Apr 2023 11:19:42 +0200 Subject: [PATCH 21/21] creo2urdf: add visual/collision meshes, fix continous joint --- src/creo2urdf/src/Creo2Urdf.cpp | 45 ++++++++++++++++++++++++++++----- 1 file changed, 39 insertions(+), 6 deletions(-) diff --git a/src/creo2urdf/src/Creo2Urdf.cpp b/src/creo2urdf/src/Creo2Urdf.cpp index 48d580d..38c8abb 100644 --- a/src/creo2urdf/src/Creo2Urdf.cpp +++ b/src/creo2urdf/src/Creo2Urdf.cpp @@ -5,6 +5,7 @@ * This software may be modified and distributed under the terms of the * BSD-3-Clause license. See the accompanying LICENSE file for details. */ +#define _USE_MATH_DEFINES #include #include @@ -37,7 +38,7 @@ enum class c2uLogLevel WARN }; -std::map log_level_key = { +const std::map log_level_key = { {c2uLogLevel::INFO, "c2uINFO"}, {c2uLogLevel::WARN, "c2uWARN"} }; @@ -47,7 +48,7 @@ void printToMessageWindow(pfcSession_ptr session, std::string message, c2uLogLev xstringsequence_ptr msg_sequence = xstringsequence::create(); msg_sequence->append(xstring(message.c_str())); session->UIClearMessage(); - session->UIDisplayMessage("creo2urdf.txt", log_level_key[log_level].c_str(), msg_sequence); + session->UIDisplayMessage("creo2urdf.txt", log_level_key.at(log_level).c_str(), msg_sequence); } @@ -198,13 +199,20 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { printToMessageWindow(session_ptr, "unit vector of axis " + string(axis->GetName()) + " is : (" + std::to_string(unit[0]) + ", " + std::to_string(unit[1]) + ", " + std::to_string(unit[2]) + ")"); - // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it - modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create(csys_list->get(0)->GetName()))); - link.setInertia(fromCreo(mass_prop)); if (i == asm_component_list->getarraysize() - 1) { // TODO This is valid only for twobars iDynTree::RevoluteJoint joint(fromCreo(transform), { {unit[0], unit[1], unit[2]}, - { o->get(0) * mm_to_m, o->get(1) * mm_to_m, o->get(2) * mm_to_m } }); + iDynTree::Position().Zero()}); + // Should be 0 the origin of the axis, the displacement is already considered in transform + //{ o->get(0) * mm_to_m, o->get(1) * mm_to_m, o->get(2) * mm_to_m } }); + + // TODO let's put the limits hardcoded, to be retrieved from Creo + double min = 0.0; + double max = M_PI; + joint.enablePosLimits(true); + joint.setPosLimits(0, min, max); + // TODO we have to retrieve the rest transform from creo + //joint.setRestTransform() if (idyn_model.addJointAndLink(prevLinkName, prevLinkName + "--" + string(name), &joint, string(name), link) == iDynTree::JOINT_INVALID_INDEX) { printToMessageWindow(session_ptr, "FAILED TO ADD JOINT!"); @@ -217,6 +225,31 @@ class Creo2UrdfActionListerner : public pfcUICommandActionListener { idyn_model.addLink(string(name), link); } + + // Getting just the first csys is a valid assumption for the MVP-1, for more complex asm we will need to change it + modelhdl->Export(name + ".stl", pfcExportInstructions::cast(pfcSTLBinaryExportInstructions().Create(csys_list->get(0)->GetName()))); + // Lets add the mess to the link + iDynTree::ExternalMesh visualMesh; + // Meshes are in millimeters, while iDynTree models are in meters + iDynTree::Vector3 scale; scale(0) = scale(1) = scale(2) = mm_to_m; + visualMesh.setScale(scale); + // Let's assign a gray as default color + iDynTree::Vector4 color; + iDynTree::Material material; + color(0) = color(1) = color(2) = 0.5; + color(3) = 1.0; + material.setColor(color); + visualMesh.setMaterial(material); + // Assign transform + // TODO Right now maybe it is not needed it ie exported respct the link csys + // visualMesh.link_H_geometry = link_H_geometry; + + // Assign name + visualMesh.setFilename(string(name) + ".stl"); + // TODO Right now let's consider visual and collision with the same mesh + idyn_model.visualSolidShapes().getLinkSolidShapes()[idyn_model.getLinkIndex(string(name))].push_back(visualMesh.clone()); + idyn_model.collisionSolidShapes().getLinkSolidShapes()[idyn_model.getLinkIndex(string(name))].push_back(visualMesh.clone()); + //idyn_model.addAdditionalFrameToLink(string(name), string(name) + "_" + string(csys_list->get(0)->GetName()), fromCreo(transform)); TODO when we have an additional frame to add }