Compare commits
20 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| c9e5a558d9 | |||
| d615530519 | |||
| 5be6539a1e | |||
| 7c7fa77608 | |||
| 5ad54c18f1 | |||
| 07b9a60a25 | |||
| 6f625c46ae | |||
| 33048785f9 | |||
| cad4045171 | |||
| b19415b030 | |||
| dd2e8f357a | |||
| 415eb4d32a | |||
| 752d307cde | |||
| 7b14c98461 | |||
| a0314f3222 | |||
| 6a60cd2b5e | |||
| 9df7f78665 | |||
| 6211c6d873 | |||
| 4b97a5b949 | |||
| eed869c997 |
@@ -48,7 +48,6 @@ KEY_SPIRAL_VASE = "SpiralVase"
|
||||
KEY_SPIRAL_VASE_LEN = 'SpiralVaseLen'
|
||||
KEY_SPIRAL_VASE_INTERP_LEN = 'SpiralVaseInterpLen'
|
||||
KEY_SLICING_TYPE = "SlicingType"
|
||||
KEY_SLICING_DIR = "SlicingDir"
|
||||
KEY_SLICE_STEP = "StrandH"
|
||||
KEY_SHELLS_NBR = "StrandCount"
|
||||
KEY_STRAND = "StrandW"
|
||||
@@ -86,6 +85,8 @@ KEY_WIPE_LEN = "WipeLen"
|
||||
KEY_WIPE_DIR = "WipeDir"
|
||||
KEY_WIPE_FEEDPU = "WipeFeedPu"
|
||||
KEY_TOOL_DIAM = "ToolDiam"
|
||||
KEY_MAX_STRANDH_FACTOR = 'MaxStrandHFactor'
|
||||
KEY_MIN_STRANDH_FACTOR = 'MinStrandHFactor'
|
||||
|
||||
-- Solid Fill
|
||||
KEY_FLOOR_NBR = "FloorCount"
|
||||
@@ -177,6 +178,7 @@ KEY_AUX_SOLIDS_WIPE_DIR = "AuxSolidsWipeDir"
|
||||
|
||||
-- Parametri di macchina
|
||||
SEC_3DPRINTING = "3dPrinting"
|
||||
KEY_HORIZ_EXTR = "HorizExtrusion"
|
||||
KEY_COEFF_X = "CoeffX"
|
||||
KEY_COEFF_Y = "CoeffY"
|
||||
KEY_SPEED_MIN = "SpeedMin"
|
||||
@@ -189,9 +191,9 @@ KEY_LINEAR_TOL = 'LinearTol'
|
||||
KEY_MAX_H = "SlicingHeight"
|
||||
KEY_MAX_SLICES_NBR = "MaxSlicesNumber"
|
||||
KEY_SLICE_NBR = "SliceNbr"
|
||||
KEY_SLICE_Z = "SliceZ"
|
||||
KEY_SLICE_DIR = "SliceDir"
|
||||
KEY_SLICE_POS = "SlicePos"
|
||||
KEY_SLICE_DELTAZ = "DeltaZ"
|
||||
KEY_SLICE_REAL_Z = "SliceRealZ"
|
||||
KEY_ZIG_ZAG_INFILL = "ZigZagInfill"
|
||||
KEY_ZIG_ZAG_DIR = "ZigZagDir"
|
||||
KEY_INVERTED_CRV = "InvertedCrv"
|
||||
@@ -215,12 +217,14 @@ KEY_CRV_OFFSET = "CurveOffset"
|
||||
KEY_ORIG_REF = "Orig"
|
||||
KEY_WIPE_ON_CRV = "WipeOnCrv"
|
||||
KEY_FEED_COEFF = 'FeedCoeff'
|
||||
KEY_ENTITY_NAME = 'EntityName'
|
||||
|
||||
SLICING_TYPE = {
|
||||
VERTICAL = 1,
|
||||
DEG45_X = 2,
|
||||
DEG45_Y = 3,
|
||||
HORIZONTAL = 4,
|
||||
MULTIPLANAR = 5,
|
||||
}
|
||||
|
||||
TYPE = {
|
||||
@@ -303,6 +307,7 @@ LAY_RIBS = "Ribs"
|
||||
LAY_AUX_SOLIDS = "AuxSolids"
|
||||
LAY_SHELL_NBR = "ShellNumber"
|
||||
LAY_AUX = "Aux"
|
||||
SPINE_CURVE = "Spine"
|
||||
LAY_FRAME = "Frame"
|
||||
VIEWPARAMS = 'ViewParams'
|
||||
IMPORTED_SOLID = 'ImportedSolid'
|
||||
|
||||
+9
-10
@@ -31,7 +31,7 @@ local function GetLayerParamsForPathCalc()
|
||||
LayerParams.dLayHeight = EgtGetInfo( s_nPartId, KEY_SLICE_STEP, 'd')
|
||||
LayerParams.dStrandOverlap = EgtGetInfo( s_nPartId, KEY_STRAND_OVERLAP, 'd') or 0
|
||||
LayerParams.dOffs = EgtGetInfo( s_nPartId, KEY_OFFSET_SLICE, 'd')
|
||||
s_vtSlicing = EgtGetInfo( s_nPartId, KEY_SLICING_DIR, 'v')
|
||||
s_vtSlicing = EgtGetInfo( s_nPartId, KEY_SLICE_DIR, 'v')
|
||||
LayerParams.bPrintInvert = ( EgtGetInfo( s_nPartId, KEY_PRINT_DIRECTION, 'i') == PRINT_DIRECTION.CW)
|
||||
LayerParams.vPrintOrder = EgtGetInfo( s_nPartId, KEY_PRINT_ORDER, 'vi') or { 1, 2, 3, 4, 5, 6, 7, 8}
|
||||
-- parametri costolature
|
||||
@@ -116,19 +116,13 @@ end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function GetLayerStartPoint( nLayId)
|
||||
|
||||
-- recupero quota layer
|
||||
local dZ = EgtGetInfo( nLayId, KEY_SLICE_REAL_Z, 'd')
|
||||
local dDeltaZ = EgtGetInfo( nLayId, KEY_SLICE_DELTAZ, 'd')
|
||||
local frSlicing = Frame3d( ORIG(), s_vtSlicing)
|
||||
local ptOn = Point3d( 0, 0, dZ + dDeltaZ)
|
||||
ptOn:toGlob( frSlicing)
|
||||
|
||||
|
||||
-- gruppo temporaneo dove salvo i risultati
|
||||
local nGrpTmp = EgtGroup( nLayId)
|
||||
|
||||
local vPtStart = {}
|
||||
local nMachStartGrp = EgtGetFirstNameInGroup( s_nPartId, LAY_MACH_START)
|
||||
local ptOn = EgtGetInfo( nLayId, KEY_SLICE_POS, 'p')
|
||||
local nStartId = EgtGetFirstInGroup( nMachStartGrp)
|
||||
while nStartId do
|
||||
local nType = EgtGetType( nStartId)
|
||||
@@ -4343,12 +4337,17 @@ function CalcPaths.Exec( nPartId)
|
||||
local LayerParams = GetLayerParamsForPathCalc()
|
||||
local nFirstSolidLay = EgtGetInfo( s_nPartId, KEY_FIRST_SOLID_LAY, 'i') or -1
|
||||
local nLastSolidLay = EgtGetInfo( s_nPartId, KEY_LAST_SOLID_LAY, 'i') or -1
|
||||
local nSlicingType = EgtGetInfo( s_nPartId, KEY_SLICING_TYPE, 'i')
|
||||
|
||||
-- scorro tutti i suoi layer
|
||||
for nIdx = 1, #vLayIds do
|
||||
|
||||
s_nCurrIdx = nIdx
|
||||
|
||||
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
s_vtSlicing = EgtGetInfo( vLayIds[nIdx], KEY_SLICE_DIR, 'v')
|
||||
end
|
||||
|
||||
local nRibsGrp = EgtGetFirstNameInGroup( vLayIds[nIdx], RIBS_GRP)
|
||||
local bIgnoreMergedRibs = true
|
||||
local nAuxSolidsGrp = EgtGetFirstNameInGroup( vLayIds[nIdx], AUX_SOLIDS_GRP)
|
||||
|
||||
+416
-158
@@ -18,38 +18,101 @@ local s_vErr = {}
|
||||
local s_dStrand
|
||||
|
||||
-- costanti
|
||||
local TOLER = 0.05
|
||||
local MID_TOLER = 0.1
|
||||
-- recupero tolleranza dal file ini
|
||||
local INI_TOLER = EgtGetNumberFromIni( 'GeomDB', 'SurfTmToler', 0.01, EgtGetIniFile())
|
||||
local MID_TOLER = 10 * INI_TOLER
|
||||
local TOLER = 0.5 * MID_TOLER
|
||||
local BIG_TOLER = 2.0
|
||||
local MIN_LEN = 20.0
|
||||
local MIN_AREA = 25.0
|
||||
local DELTAZ = 10 * GEO.EPS_SMALL
|
||||
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function ComputeZSlices( dSliceStep, dZmin, dDeltaZ, dZmax)
|
||||
local function ComputeSlicingData( vIds, vtSlicing, dSliceStep, dMaxH, nMaxSlicesNbr)
|
||||
-- la direzione di slicing è vtSlicing per tutti gli slices
|
||||
-- i valori di slicing sono le distanze dall'origine dei piani di slicing misurate in direzione vtSlicing
|
||||
|
||||
-- verifico se è impostato un numero massimo di slices
|
||||
local nMaxSlicesNbr = EgtGetInfo( s_nPartId, KEY_MAX_SLICES_NBR, 'i') or 0
|
||||
if nMaxSlicesNbr == 0 then
|
||||
nMaxSlicesNbr = GEO.INFINITO -- da gestire come caso illimitato
|
||||
local vSlicingDir = { vtSlicing}
|
||||
-- ricavo le z a cui calcolare gli slices
|
||||
local frSlicing = Frame3d( ORIG(), vtSlicing)
|
||||
local b3Box = BBox3d()
|
||||
for i = 1, #vIds do
|
||||
local b3Tmp = EgtGetBBoxRef( vIds[i], GDB_BB.STANDARD, frSlicing)
|
||||
b3Box:Add( b3Tmp)
|
||||
end
|
||||
|
||||
local dZmin = b3Box:getMin():getZ()
|
||||
local dZmax = min( b3Box:getMax():getZ(), dMaxH)
|
||||
|
||||
local nCnt = 1
|
||||
local ptSlices = { dZmin + dDeltaZ}
|
||||
local vZSlices = { dZmin + DELTAZ}
|
||||
local dPosZ = dZmin + dSliceStep
|
||||
while dPosZ < dZmax and nCnt < nMaxSlicesNbr do
|
||||
table.insert( ptSlices, dPosZ)
|
||||
table.insert( vZSlices, dPosZ)
|
||||
dPosZ = dPosZ + dSliceStep
|
||||
nCnt = nCnt + 1
|
||||
end
|
||||
return ptSlices
|
||||
end
|
||||
return vZSlices, vSlicingDir
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function ComputeMultiplanarSlicingData( dSliceStep, nMaxSlicesNbr)
|
||||
-- le direzioni e i valori di slicing sono le normali e i punti che definiscono i piani di slicing e vengono calcolati sulla curva "Spine"
|
||||
|
||||
-- recupero la curva guida ( è la prima nel gruppo aux con info NameEntity = Spine oppure quella con nome Spine)
|
||||
local nAuxLayId = EgtGetFirstNameInGroup( s_nPartId, LAY_AUX)
|
||||
local vIds = EgtGetAllInGroup( nAuxLayId)
|
||||
local nSpineId
|
||||
for i = 1, #vIds do
|
||||
local sNameEntity = EgtGetInfo( vIds[i], KEY_ENTITY_NAME)
|
||||
if sNameEntity == SPINE_CURVE then
|
||||
nSpineId = vIds[i]
|
||||
break
|
||||
elseif EgtGetName( vIds[i]) == SPINE_CURVE then
|
||||
nSpineId = vIds[i]
|
||||
end
|
||||
end
|
||||
if not nSpineId or ( EgtGetType( nSpineId) & GDB_FY.GEO_CURVE) == 0 then
|
||||
EgtOutBox( "No spine for multiplanar slicing", 'Error', 'ERROR')
|
||||
return nil
|
||||
end
|
||||
|
||||
-- verifico che la spina parta nell'origine con direzione Z_AX
|
||||
local ptS = EgtSP( nSpineId, GDB_ID.ROOT)
|
||||
local vtS = EgtSV( nSpineId, GDB_ID.ROOT)
|
||||
if abs( ptS:getZ()) > GEO.EPS_SMALL then
|
||||
EgtOutLog( 'Error : spine does not start on table')
|
||||
EgtOutBox( "Spine not valid for multiplanar slicing", 'Error', 'ERROR')
|
||||
return nil
|
||||
elseif not AreSameVectorApprox( vtS, Z_AX()) then
|
||||
EgtOutLog( 'Error : spine start direction is not vertical')
|
||||
EgtOutBox( "Spine not valid for multiplanar slicing", 'Error', 'ERROR')
|
||||
return nil
|
||||
end
|
||||
|
||||
local vSlicingDir = {}
|
||||
local vSlicingPnt = {}
|
||||
-- ciclo sulla curva
|
||||
local nCnt = 0
|
||||
local dCrvLen = EgtCurveLength( nSpineId)
|
||||
for i = 0, nMaxSlicesNbr - 1 do
|
||||
local dCurrLen = i * dSliceStep
|
||||
if dCurrLen > dCrvLen then
|
||||
break
|
||||
end
|
||||
local dU = EgtCurveParamAtLength( nSpineId, dCurrLen)
|
||||
vSlicingDir[i+1] = EgtUV( nSpineId, dU, -1, GDB_ID.ROOT)
|
||||
vSlicingPnt[i+1] = EgtUP( nSpineId, dU, GDB_ID.ROOT)
|
||||
end
|
||||
|
||||
return vSlicingPnt, vSlicingDir
|
||||
end
|
||||
--------------------------------------------------------------------
|
||||
local function ComputeMaxH( vIds, frSlicing, dSliceStep)
|
||||
local function ComputeMaxH( vIds, vtSlicing, dSliceStep)
|
||||
|
||||
local HMax = EgtGetInfo( s_nPartId, KEY_MAX_H, 'd') or GEO.INFINITO
|
||||
if HMax < GEO.EPS_SMALL then
|
||||
HMax = GEO.INFINITO
|
||||
return GEO.INFINITO
|
||||
end
|
||||
|
||||
-- calcolo il box globale
|
||||
@@ -60,14 +123,16 @@ local function ComputeMaxH( vIds, frSlicing, dSliceStep)
|
||||
end
|
||||
EgtSetInfo( s_nPartId, KEY_BOX_MIN_Z, b3BoxGlob:getMin():getZ())
|
||||
|
||||
if b3BoxGlob:getMax():getZ() < b3BoxGlob:getMin():getZ() + HMax + GEO.EPS_SMALL then
|
||||
-- se il pezzo non è più alto della quota massima, posso ignorarla
|
||||
if b3BoxGlob:getMax():getZ() < HMax + GEO.EPS_SMALL then
|
||||
return GEO.INFINITO
|
||||
end
|
||||
|
||||
if AreSameVectorApprox( frSlicing:getVersZ(), Z_AX()) then
|
||||
if AreSameVectorApprox( vtSlicing, Z_AX()) then
|
||||
-- se slicing verticale o multiplanare l'altezza è quella letta
|
||||
return HMax
|
||||
else
|
||||
-- se slicing inclinato
|
||||
else
|
||||
-- se slicing inclinato calcolo l'altezza massima rispetto alle quote di slicing
|
||||
local nGrpTmp = EgtGroup( s_nPartId)
|
||||
for i = 1, #vIds do
|
||||
EgtPlaneSurfTmInters( Point3d( 0, 0, HMax), Z_AX(), vIds[i], nGrpTmp, GDB_RT.GLOB)
|
||||
@@ -81,8 +146,9 @@ local function ComputeMaxH( vIds, frSlicing, dSliceStep)
|
||||
local ptMax = b3Box:getMax()
|
||||
-- la quota dell'ultimo slice è più bassa di dSliceStep rispetto alla quota dell'ultimo toolpath
|
||||
-- ( andrebbe anche considerata dCorrZ)
|
||||
ptMin = ptMin - dSliceStep * frSlicing:getVersZ()
|
||||
ptMax = ptMax - dSliceStep * frSlicing:getVersZ()
|
||||
ptMin = ptMin - dSliceStep * vtSlicing
|
||||
ptMax = ptMax - dSliceStep * vtSlicing
|
||||
local frSlicing = Frame3d( ORIG(), vtSlicing)
|
||||
ptMin:toLoc( frSlicing)
|
||||
ptMax:toLoc( frSlicing)
|
||||
|
||||
@@ -92,7 +158,9 @@ local function ComputeMaxH( vIds, frSlicing, dSliceStep)
|
||||
return GEO.INFINITO
|
||||
end
|
||||
|
||||
--------------------------------------------------------------------
|
||||
----------------------------------------------------------------------
|
||||
-------------------- SLICING EXTRA OBJECTS ---------------------------
|
||||
----------------------------------------------------------------------
|
||||
local function ReadParam( nId, sKey, sType, defVal, table)
|
||||
-- verifico se info nell'oggetto specifico
|
||||
local info = EgtGetInfo( nId, sKey, sType)
|
||||
@@ -177,13 +245,12 @@ local function GetAuxSolidsParams( nId)
|
||||
return AuxSolidsParam
|
||||
end
|
||||
|
||||
----------------------------------------------------------------------
|
||||
-------------------- SLICING EXTRA OBJECTS ---------------------------
|
||||
----------------------------------------------------------------------
|
||||
local function SliceStm( vIds, nLayId, vtSlicing, nType, sName, sNameGrp, tabParams)
|
||||
--------------------------------------------------------------------
|
||||
local function SliceStm( vIds, nLayId, vtSlicing, ptSlicing, nType, sName, sNameGrp, tabParams)
|
||||
|
||||
if not vIds or #vIds == 0 then return end
|
||||
|
||||
local nLayCnt = EgtGetInfo( nLayId, KEY_SLICE_NBR, 'i')
|
||||
-- recupero il gruppo dove salvare lo slicing degli oggetti
|
||||
local nGrp = EgtGetFirstNameInGroup( nLayId, sNameGrp)
|
||||
if not nGrp then
|
||||
@@ -192,22 +259,17 @@ local function SliceStm( vIds, nLayId, vtSlicing, nType, sName, sNameGrp, tabPar
|
||||
EgtSetName( nGrp, sNameGrp)
|
||||
EgtSetStatus( nGrp, GDB_ST.OFF)
|
||||
end
|
||||
|
||||
|
||||
-- recupero quota per slicing
|
||||
local dZ = EgtGetInfo( nLayId, KEY_SLICE_REAL_Z, 'd')
|
||||
local dDeltaZ = EgtGetInfo( nLayId, KEY_SLICE_DELTAZ, 'd')
|
||||
local nLayCnt = EgtGetInfo( nLayId, KEY_SLICE_NBR, 'i')
|
||||
|
||||
-- gruppo temporaneo per approx curve
|
||||
local frloc = Frame3d( ORIG(), vtSlicing)
|
||||
local nGrpTmp = EgtGroup( nLayId, frloc, GDB_RT.GLOB)
|
||||
local ptOn = ptSlicing + dDeltaZ * vtSlicing
|
||||
|
||||
for i = 1, #vIds do
|
||||
-- verifico che oggetto sia trimesh
|
||||
local nTypeObj = EgtGetType( vIds[i])
|
||||
if nTypeObj == GDB_TY.SRF_MESH then
|
||||
-- slicing oggetto
|
||||
local nNewId, nPntCnt, nCrvCnt, nSrfCnt = EgtPlaneSurfTmInters( ORIG() + ( dZ + dDeltaZ) * vtSlicing, vtSlicing, vIds[i], nGrp, GDB_RT.GLOB, TOLER)
|
||||
local nNewId, nPntCnt, nCrvCnt, nSrfCnt = EgtPlaneSurfTmInters( ptOn, vtSlicing, vIds[i], nGrp, GDB_RT.GLOB, TOLER)
|
||||
|
||||
-- verifico se necessario ricalcolo
|
||||
local dCorr = 0
|
||||
@@ -228,7 +290,7 @@ local function SliceStm( vIds, nLayId, vtSlicing, nType, sName, sNameGrp, tabPar
|
||||
EgtErase( j)
|
||||
end
|
||||
dCorr = 0.01
|
||||
nNewId, nPntCnt, nCrvCnt, nSrfCnt = EgtPlaneSurfTmInters( ORIG() + ( dZ + dDeltaZ + dCorr) * vtSlicing, vtSlicing, vIds[i], nGrp, GDB_RT.GLOB, TOLER)
|
||||
nNewId, nPntCnt, nCrvCnt, nSrfCnt = EgtPlaneSurfTmInters( ptOn + dCorr * vtSlicing, vtSlicing, vIds[i], nGrp, GDB_RT.GLOB, TOLER)
|
||||
end
|
||||
|
||||
if nNewId then
|
||||
@@ -251,7 +313,7 @@ local function SliceStm( vIds, nLayId, vtSlicing, nType, sName, sNameGrp, tabPar
|
||||
end
|
||||
-- rinomino le curve, correggo di DeltaZ e assegno parametri
|
||||
for nId = nChainId, nChainId + nCnt - 1 do
|
||||
EgtSetName( nId, sName .. tostring( i))
|
||||
EgtSetName( nId, sName .. EgtNumToString( i))
|
||||
EgtMove( nId, - ( dDeltaZ + dCorr) * vtSlicing)
|
||||
EgtSetInfo( nId, KEY_ORIGINAL_SURF, vIds[i])
|
||||
if tabParams then
|
||||
@@ -260,9 +322,7 @@ local function SliceStm( vIds, nLayId, vtSlicing, nType, sName, sNameGrp, tabPar
|
||||
end
|
||||
end
|
||||
EgtModifyCurveExtrusion( nId, vtSlicing, GDB_RT.GLOB)
|
||||
EgtRelocateGlob( nId, nGrpTmp)
|
||||
EgtApproxCurve( nId, GDB_CA.ARCS, MID_TOLER)
|
||||
EgtRelocateGlob( nId, nGrp)
|
||||
|
||||
-- se ho ancora curve aperte, segnalo errore
|
||||
if nType ~= TYPE.RIB and nType ~= TYPE.INFILL and not EgtCurveIsClosed( nId) then
|
||||
@@ -282,12 +342,14 @@ local function SliceStm( vIds, nLayId, vtSlicing, nType, sName, sNameGrp, tabPar
|
||||
-- verifico che il gruppo non sia vuoto
|
||||
if not EgtGetFirstInGroup( nGrp) then
|
||||
EgtErase( nGrp)
|
||||
nGrp = nil
|
||||
end
|
||||
EgtErase( nGrpTmp)
|
||||
|
||||
return nGrp
|
||||
end
|
||||
|
||||
--------------------------------------------------------------------
|
||||
local function SlicingExtraObjects( nLay, vtSlicing, nType, sName, sNameGrp, nStmId)
|
||||
local function SlicingExtraObjects( nLay, nType, sName, sNameGrp, nStmId)
|
||||
|
||||
if not nLay then return end
|
||||
|
||||
@@ -340,19 +402,20 @@ local function SlicingExtraObjects( nLay, vtSlicing, nType, sName, sNameGrp, nSt
|
||||
-- per ogni layer calcolo lo slicing degli oggetti
|
||||
local vLayIds = EgtGetNameInGroup( s_nPartId, SLICE_LAYER .. '*')
|
||||
for i = 1, #vLayIds do
|
||||
SliceStm( vSliceIds, vLayIds[i], vtSlicing, nType, sName, sNameGrp, tabParams)
|
||||
-- ricavo piano di slicing
|
||||
local ptSlicing = EgtGetInfo( vLayIds[i], KEY_SLICE_POS, 'p')
|
||||
local vtSlicing = EgtGetInfo( vLayIds[i], KEY_SLICE_DIR, 'v')
|
||||
SliceStm( vSliceIds, vLayIds[i], vtSlicing, ptSlicing, nType, sName, sNameGrp, tabParams)
|
||||
end
|
||||
|
||||
-- rimuovo eventuali oggetti creati per lo slicing
|
||||
for i = 1, #vEraseIds do
|
||||
EgtErase( vEraseIds[i])
|
||||
end
|
||||
EgtErase( vEraseIds)
|
||||
|
||||
return true
|
||||
end
|
||||
|
||||
------------------------------------------------------------------------
|
||||
local function SlicingInfill( nLay, vtSlicing, sName, sNameGrp)
|
||||
local function SlicingInfill( nLay, sName, sNameGrp)
|
||||
|
||||
-- recupero tutte le curve di infill suddivise per direzioni di riferimento
|
||||
local vInfillGrps = EgtGetAllInGroup( nLay)
|
||||
@@ -362,10 +425,6 @@ local function SlicingInfill( nLay, vtSlicing, sName, sNameGrp)
|
||||
table.insert( tIds, vIds)
|
||||
end
|
||||
local nGrps = #vInfillGrps
|
||||
|
||||
-- recupero quota infill
|
||||
local ptInfill = EgtSP( tIds[1][1], GDB_ID.ROOT)
|
||||
local dZInfill = ( ptInfill - ORIG()) * vtSlicing
|
||||
|
||||
local vLayIds = EgtGetNameInGroup( s_nPartId, SLICE_LAYER .. '*')
|
||||
for i = 1, #vLayIds do
|
||||
@@ -373,21 +432,14 @@ local function SlicingInfill( nLay, vtSlicing, sName, sNameGrp)
|
||||
-- individuo il gruppo di cui fare slicing
|
||||
local nGrpIdx = ( i - 1) % nGrps + 1
|
||||
|
||||
-- recupero il gruppo dove salvare gli infill
|
||||
local nGrp = EgtGetFirstNameInGroup( vLayIds[i], sNameGrp)
|
||||
if not nGrp then
|
||||
-- se gruppo non esiste lo creo
|
||||
nGrp = EgtGroup( vLayIds[i])
|
||||
EgtSetName( nGrp, sNameGrp)
|
||||
EgtSetStatus( nGrp, GDB_ST.OFF)
|
||||
end
|
||||
|
||||
-- copio le curve di infill e le traslo alla quota dello slicing
|
||||
local dZ = EgtGetInfo( vLayIds[i], KEY_SLICE_REAL_Z, 'd')
|
||||
for j = 1, #tIds[nGrpIdx] do
|
||||
local nCrv = EgtCopyGlob( tIds[nGrpIdx][j], nGrp)
|
||||
EgtSetName( nCrv, sName .. tostring(j))
|
||||
EgtMove( nCrv, (dZ - dZInfill) * vtSlicing, GDB_RT.GLOB)
|
||||
-- recupero i dati di slicing
|
||||
local ptSlicing = EgtGetInfo( vLayIds[i], KEY_SLICE_POS, 'p')
|
||||
local vtSlicing = EgtGetInfo( vLayIds[i], KEY_SLICE_DIR, 'v')
|
||||
local nResultLay = SliceStm( tIds[nGrpIdx], vLayIds[i], vtSlicing, ptSlicing, TYPE.INFILL, sName, sNameGrp)
|
||||
-- sistemo i nomi delle curve
|
||||
local vInfillCrvs = EgtGetAllInGroup( nResultLay)
|
||||
for j = 1, #vInfillCrvs do
|
||||
EgtSetName( vInfillCrvs[j], sName .. EgtNumToString(j))
|
||||
end
|
||||
end
|
||||
|
||||
@@ -548,7 +600,7 @@ local function CalcGridFromPattern( nInfillGrp, nStmId, dOffsStm, frRef, vtOffs,
|
||||
EgtAddCurveCompoLine( nCrvBase, pt2)
|
||||
end
|
||||
CompletePatternBaseCrv( nCrvBase, dAng, dDim1, dDim2, ptMin:getY(), ptMax:getY())
|
||||
|
||||
|
||||
-- preparo le altre curve
|
||||
CalcPatternCrvs( nMainGrp, nCrvBase, ptMin:getX(), ptMax:getX(), dAng, dDim1, dStrand)
|
||||
|
||||
@@ -586,7 +638,7 @@ end
|
||||
--------------------------------------------------------------------
|
||||
local function CalcInfill( nInfillGrp, nType, dDensity, dDir, dOffsX, dOffsY, dGridOverlap, nStmId, dOffsStm, dStrand, vtSlicing)
|
||||
|
||||
-- frame locale alla direzione dell'infill
|
||||
-- frame locale alla direzione dell'infill
|
||||
local frLoc = Frame3d( ORIG(), vtSlicing)
|
||||
frLoc:rotate( ORIG(), vtSlicing, dDir)
|
||||
|
||||
@@ -626,6 +678,21 @@ local function CalcInfill( nInfillGrp, nType, dDensity, dDir, dOffsX, dOffsY, dG
|
||||
local vSlicesAng = EgtIf( nType == FILL_TYPE.HONEYCOMB, {0}, {0, 60, 120})
|
||||
CalcGridFromPattern( nInfillGrp, nStmId, dOffsStm, frLoc, vtOffs, dDim1, dDim2, dAng, vSlicesAng, dRealStrand, vtSlicing)
|
||||
end
|
||||
|
||||
-- calcolo le superifici a partire dalle curve ( nei casi standard è semplice estrusione)
|
||||
local vGrps = EgtGetAllInGroup( nInfillGrp)
|
||||
local vCrvs = {}
|
||||
for i = 1, #vGrps do
|
||||
EgtJoinTables( vCrvs, EgtGetAllInGroup( vGrps[i]))
|
||||
end
|
||||
local nLastLay = EgtGetLastNameInGroup( s_nPartId, SLICE_LAYER .. '*')
|
||||
local ptLast = EgtGetInfo( nLastLay, KEY_SLICE_POS, 'p')
|
||||
local dZ = ( ptLast - ORIG()) * vtSlicing
|
||||
for i = 1, #vCrvs do
|
||||
EgtSurfTmByExtrusion( EgtGetParent( vCrvs[i]), vCrvs[i], ( dZ + 100) * vtSlicing, GDB_RT.GLOB)
|
||||
end
|
||||
EgtErase( vCrvs)
|
||||
|
||||
end
|
||||
|
||||
--------------------------------------------------------------------
|
||||
@@ -660,7 +727,7 @@ local function PrepareInfill( nStmId, vtSlicing)
|
||||
CalcInfill( nInfillGrp, nType, dDensity, dDir, dOffsX, dOffsY, dGridOverlap, nStmId, dOffsStm, dStrand, vtSlicing)
|
||||
|
||||
-- aggiungo allo slicing
|
||||
SlicingInfill( nInfillGrp, vtSlicing, INFILL_CRV, INFILL_GRP)
|
||||
SlicingInfill( nInfillGrp, INFILL_CRV, INFILL_GRP)
|
||||
end
|
||||
|
||||
----------------------------------------------------------------------
|
||||
@@ -689,15 +756,15 @@ local function PrepareAuxSolidsInfill( nSolidsLay, vtSlicing)
|
||||
|
||||
-- creo gruppo associato
|
||||
local nInfillGrp = EgtGroup( s_nPartId)
|
||||
EgtSetName( nInfillGrp, AUX_SOLIDS_INFILL_GRP .. tostring( vIds[i]))
|
||||
EgtSetName( nInfillGrp, AUX_SOLIDS_INFILL_GRP .. EgtNumToString( vIds[i]))
|
||||
EgtSetStatus( nInfillGrp, GDB_ST.OFF)
|
||||
|
||||
-- calcolo infill
|
||||
CalcInfill( nInfillGrp, nType, dDensity, dDir, dOffsX, dOffsY, dGridOverlap, vIds[i], 0, dStrand, vtSlicing)
|
||||
|
||||
-- aggiungo allo slicing
|
||||
local sName = AUX_SOLIDS_INFILL_CRV .. tostring( vIds[i]) .. '_'
|
||||
SlicingInfill( nInfillGrp, vtSlicing, sName, AUX_SOLIDS_GRP)
|
||||
local sName = AUX_SOLIDS_INFILL_CRV .. EgtNumToString( vIds[i]) .. '_'
|
||||
SlicingInfill( nInfillGrp, sName, AUX_SOLIDS_GRP)
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -725,7 +792,7 @@ local function ExtractRibsLoops( nRibsGrp, nStmId)
|
||||
if nCrv then
|
||||
-- assegno nome che permetta di ricondurli alla superficie da cui derivano
|
||||
for nId = nCrv, nCrv + nCnt - 1 do
|
||||
EgtSetName( nId, SURF_LOOP .. tostring( vIds[i]))
|
||||
EgtSetName( nId, SURF_LOOP .. EgtNumToString( vIds[i]))
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -743,50 +810,191 @@ local function ValueInArray( vArr, nValue)
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function SlicingNoSolid( nRibsLay, vZSlices, dDeltaZStart, dZmin, frSlicing)
|
||||
|
||||
local vtSlicing = frSlicing:getVersZ()
|
||||
|
||||
local function SlicingNoSolid( nRibsLay, nSlicingType, dMaxH, dSliceStep, vSlicingVal, vSlicingDir)
|
||||
|
||||
-- contatore per il calcolo delle intersezioni
|
||||
if EgtProcessEvents( 100, 0) == 1 then return false end
|
||||
|
||||
-- creo i layer dello slicing
|
||||
for i = 1, #vZSlices do
|
||||
local nLayGrp = EgtGroup( s_nPartId)
|
||||
EgtSetName( nLayGrp, SLICE_LAYER .. tostring( i))
|
||||
|
||||
local dDeltaZ = EgtIf( i == 1, dDeltaZStart, 0)
|
||||
local dPosZ = vZSlices[i]- dDeltaZ
|
||||
EgtSetInfo( nLayGrp, KEY_SLICE_Z, dPosZ - dZmin)
|
||||
EgtSetInfo( nLayGrp, KEY_SLICE_REAL_Z, dPosZ)
|
||||
EgtSetInfo( nLayGrp, KEY_SLICE_DELTAZ, dDeltaZ)
|
||||
EgtSetInfo( nLayGrp, KEY_SLICE_NBR, i)
|
||||
|
||||
local nGrpCrv = EgtGroup( nLayGrp)
|
||||
EgtSetName( nGrpCrv, CONTOUR_GRP .. EgtNumToString( 0))
|
||||
-- recupero i setti di cui fare lo slicing e i parametri associati
|
||||
local vRibIds = EgtGetAllInGroup( nRibsLay)
|
||||
local tabParams = {}
|
||||
local vIds = {}
|
||||
for i = 1, #vRibIds do
|
||||
if EgtGetType( vRibIds[i]) == GDB_TY.SRF_MESH then
|
||||
local vParams = GetRibParams( vRibIds[i])
|
||||
if vParams[KEY_RIBS_SHELLS_NBR] > 0 then
|
||||
table.insert( tabParams, vParams)
|
||||
table.insert( vIds, vRibIds[i])
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
-- creo i layer dello slicing
|
||||
for i = 1, #vSlicingVal do
|
||||
-- creo layer e gruppo dei contorni
|
||||
local nLayId = EgtGroup( s_nPartId)
|
||||
EgtSetName( nLayId, SLICE_LAYER .. EgtNumToString( i))
|
||||
EgtSetInfo( nLayId, KEY_SLICE_NBR, i)
|
||||
local nCrvLayId = EgtGroup( nLayId)
|
||||
EgtSetName( nCrvLayId, CONTOUR_GRP .. EgtNumToString( 0))
|
||||
|
||||
-- recupero i dati del piano di slicing
|
||||
local dDeltaZ = EgtIf( i == 1, DELTAZ, 0)
|
||||
local ptSlicing
|
||||
local vtSlicing
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
vtSlicing = vtSlicing[i]
|
||||
ptSlicing = vSlicingVal[i]
|
||||
else
|
||||
vtSlicing = vSlicingDir[1]
|
||||
ptSlicing = ORIG() + ( vSlicingVal[i] - dDeltaZ) * vtSlicing
|
||||
end
|
||||
EgtSetInfo( nLayId, KEY_SLICE_POS, ptSlicing)
|
||||
EgtSetInfo( nLayId, KEY_SLICE_DIR, vtSlicing)
|
||||
EgtSetInfo( nLayId, KEY_SLICE_DELTAZ, dDeltaZ)
|
||||
|
||||
-- slicing dei setti
|
||||
local nResultLay = SliceStm( vIds, nLayId, vtSlicing, ptSlicing, TYPE.RIB, RIBS_CRV, RIBS_GRP, tabParams)
|
||||
|
||||
-- se slicing multplanare verifico altezza massima
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR and dMaxH < GEO.INFINITO then
|
||||
local b3Box = EgtGetBBoxGlob( nResultLay, GDB_BB.STANDARD)
|
||||
local ptMax = b3Box:getMax() + dSliceStep * vtSlicing
|
||||
if ptMax:getZ() > dMaxH + GEO.EPS_SMALL then
|
||||
EgtErase( nLayId)
|
||||
break
|
||||
end
|
||||
end
|
||||
|
||||
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + i / #vSlicingVal * 100, 0) == 1 then return false end
|
||||
end
|
||||
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + 20, 0) == 1 then return false end
|
||||
|
||||
-- slicing dei setti
|
||||
SlicingExtraObjects( nRibsLay, vtSlicing, TYPE.RIB, RIBS_CRV, RIBS_GRP)
|
||||
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + 100, 0) == 1 then return false end
|
||||
|
||||
return true
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function SlicingWithSolid( nStmId, vZSlices, dDeltaZStart, dZmin, frSlicing)
|
||||
local function MultiPlanarSlicing( nStmId, dMaxH, vSlicingDir, vSlicingPnt)
|
||||
|
||||
local vtSlicing = frSlicing:getVersZ()
|
||||
local dSliceStep = EgtGetInfo( s_nPartId, KEY_SLICE_STEP, 'd')
|
||||
-- ciclo sui piani di slicing calcolati
|
||||
local nFirstGrpId
|
||||
local nCnt = 0
|
||||
for i = 1, #vSlicingPnt do
|
||||
local vtN = vSlicingDir[i]
|
||||
local ptOn = vSlicingPnt[i]
|
||||
-- correzione per layer iniziale
|
||||
if i == 1 then
|
||||
ptOn = ptOn + DELTAZ * vtN
|
||||
end
|
||||
|
||||
local nGrpId = EgtGroup( s_nPartId)
|
||||
if not nFirstGrpId then
|
||||
nFirstGrpId = nGrpId
|
||||
end
|
||||
local nEntId = EgtPlaneSurfTmInters( ptOn, vtN, nStmId, nGrpId, GDB_RT.GLOB, TOLER)
|
||||
-- eventuale verifica con altezza massima
|
||||
if dMaxH < GEO.INFINITO then
|
||||
local b3Box = EgtGetBBoxGlob( nGrpId, GDB_BB.STANDARD)
|
||||
local ptMax = b3Box:getMax() + dSliceStep * vtN
|
||||
if ptMax:getZ() > dMaxH + GEO.EPS_SMALL then
|
||||
EgtErase( nGrpId)
|
||||
break
|
||||
end
|
||||
end
|
||||
nCnt = i + 1
|
||||
-- aggiornamento progress ( per riconoscimento corretto della fase dal programma deve partire da 1)
|
||||
if EgtProcessEvents( max( 1, 100 * i / #vSlicingPnt), 0) == 1 then
|
||||
return nFirstGrpId, -1
|
||||
end
|
||||
end
|
||||
return nFirstGrpId, nCnt
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function VerifyMultiPlanarStrand()
|
||||
|
||||
local vLayers = EgtGetNameInGroup( s_nPartId, SLICE_LAYER .. '*')
|
||||
-- recupero i valori di strand massimo e minimo dal file ini
|
||||
local sMachIni = EgtGetCurrMachineDir() .. '\\' .. EgtGetCurrMachineName() .. '.ini'
|
||||
local dMaxStrandFactor = EgtGetNumberFromIni( '3dPrinting', KEY_MAX_STRANDH_FACTOR, 2, sMachIni)
|
||||
local dMinStrandFactor = EgtGetNumberFromIni( '3dPrinting', KEY_MIN_STRANDH_FACTOR, 0.5, sMachIni)
|
||||
local dSliceStep = EgtGetInfo( s_nPartId, KEY_SLICE_STEP, 'd')
|
||||
local dMaxStrand = dMaxStrandFactor * dSliceStep
|
||||
local dMinStrand = dMinStrandFactor * dSliceStep
|
||||
|
||||
if dMaxStrand < dMinStrand + GEO.EPS_SMALL then
|
||||
EgtOutBox( 'Max strand height is less than min strand height', 'Error', 'ERROR')
|
||||
return false
|
||||
end
|
||||
|
||||
-- ciclo sui piani di slicing calcolati
|
||||
local vtSlicePrev = EgtGetInfo( vLayers[1], KEY_SLICE_DIR, 'v')
|
||||
local ptSlicePrev = EgtGetInfo( vLayers[1], KEY_SLICE_POS, 'p') + dSliceStep * vtSlicePrev
|
||||
for i = 2, #vLayers do
|
||||
|
||||
-- recupero il piano corrente
|
||||
local vtSlice = EgtGetInfo( vLayers[i], KEY_SLICE_DIR, 'v')
|
||||
local ptSlice = EgtGetInfo( vLayers[i], KEY_SLICE_POS, 'p') + dSliceStep * vtSlice
|
||||
local dCosAng = vtSlicePrev * vtSlice
|
||||
|
||||
-- recupero tutte le curve da verificare :
|
||||
local vCrvs = {}
|
||||
-- solido
|
||||
local vCrvGrps = EgtGetNameInGroup( vLayers[i], CONTOUR_GRP .. '*')
|
||||
for j = 1, #vCrvGrps do
|
||||
local vSolidCrvs = EgtGetNameInGroup( vCrvGrps[j], OUTER_CRV)
|
||||
EgtJoinTables( vCrvs, vSolidCrvs)
|
||||
end
|
||||
-- setti
|
||||
local nRibsGrp = EgtGetFirstNameInGroup( vLayers[i], RIBS_GRP) or GDB_ID.NULL
|
||||
local vRibsCrv = EgtGetNameInGroup( nRibsGrp, RIBS_CRV .. '*') or {}
|
||||
EgtJoinTables( vCrvs, vRibsCrv)
|
||||
-- AuxSolids
|
||||
local nAuxSolidsGrp = EgtGetFirstNameInGroup( vLayers[i], AUX_SOLIDS_GRP) or GDB_ID.NULL
|
||||
local vAuxSolidsCrv = EgtGetNameInGroup( nAuxSolidsGrp, AUX_SOLIDS_CRV .. '*') or {}
|
||||
EgtJoinTables( vCrvs, vAuxSolidsCrv)
|
||||
|
||||
-- verifico la distanza dal piano precedente lungo vtSlice
|
||||
for j = 1, #vCrvs do
|
||||
local _, dE = EgtCurveDomain( vCrvs[j])
|
||||
for dU = 0, dE do
|
||||
local ptCurr = EgtUP( vCrvs[j], dU, GDB_ID.ROOT) + dSliceStep * vtSlice
|
||||
local dDist = ( ptCurr - ptSlicePrev) * vtSlicePrev / dCosAng
|
||||
if dDist < dMinStrand - GEO.EPS_SMALL or dDist > dMaxStrand + GEO.EPS_SMALL then
|
||||
EgtOutLog( 'Error on layer ' .. EgtNumToString( i) .. ' : strand height (' .. EgtNumToString( dDist) .. ') outside valid range (' .. EgtNumToString( dMinStrand) ..
|
||||
' - ' .. EgtNumToString( dMaxStrand) .. ')')
|
||||
EgtOutBox( 'Strand height value ' .. EgtNumToString( dDist) .. ' in layer ' .. EgtNumToString( i) .. ' is outside valid range (' .. EgtNumToString( dMinStrand) ..
|
||||
' - ' .. EgtNumToString( dMaxStrand) .. ')', 'Error', 'ERROR')
|
||||
return false
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
-- aggiorno per layer successivo
|
||||
vtSlicePrev = vtSlice
|
||||
ptSlicePrev = ptSlice
|
||||
end
|
||||
|
||||
return true
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function SlicingWithSolid( nStmId, nSlicingType, dMaxH, vSlicingVal, vSlicingDir)
|
||||
|
||||
-- calcolo delle intersezioni
|
||||
if EgtProcessEvents( 1, 0) == 1 then return false end
|
||||
local nLayId = EgtParPlanesSurfTmInters( ORIG(), vtSlicing, vZSlices, nStmId, s_nPartId, GDB_RT.GLOB, TOLER)
|
||||
if EgtProcessEvents( 100, 0) == 1 then return false end
|
||||
local nLayId, nCnt
|
||||
if nSlicingType ~= SLICING_TYPE.MULTIPLANAR then
|
||||
if EgtProcessEvents( 1, 0) == 1 then return false end
|
||||
nLayId, nCnt = EgtParPlanesSurfTmInters( ORIG(), vSlicingDir[1], vSlicingVal, nStmId, s_nPartId, GDB_RT.GLOB, TOLER)
|
||||
if EgtProcessEvents( 100, 0) == 1 then return false end
|
||||
else
|
||||
nLayId, nCnt = MultiPlanarSlicing( nStmId, dMaxH, vSlicingDir, vSlicingVal)
|
||||
if nCnt == -1 then return false end
|
||||
end
|
||||
|
||||
local vPrevCen = {}
|
||||
local nLayCnt = 1
|
||||
local nCounterTot = #vZSlices + 12
|
||||
local nCounterTot = nCnt + 12
|
||||
|
||||
local nFirstSolidLay -- primo layer che contiene il solido
|
||||
local nLastSolidLay -- ultimo layer che contiene il solido
|
||||
@@ -794,12 +1002,25 @@ local function SlicingWithSolid( nStmId, vZSlices, dDeltaZStart, dZmin, frSlicin
|
||||
-- scorro i risultati dello slicing
|
||||
while nLayId do
|
||||
|
||||
local dDeltaZ = EgtIf( nLayCnt == 1, dDeltaZStart, 0)
|
||||
local dPosZ = vZSlices[nLayCnt]- dDeltaZ
|
||||
EgtSetInfo( nLayId, KEY_SLICE_Z, dPosZ - dZmin)
|
||||
EgtSetInfo( nLayId, KEY_SLICE_REAL_Z, dPosZ)
|
||||
EgtSetInfo( nLayId, KEY_SLICE_NBR, nLayCnt)
|
||||
|
||||
-- ricavo punto e direzione del piano di slicing
|
||||
local dDeltaZ = EgtIf( nLayCnt == 1, DELTAZ, 0)
|
||||
local ptSlicing
|
||||
local vtSlicing
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
-- nel caso multiplanare sono i dati di slicing
|
||||
vtSlicing = vSlicingDir[nLayCnt]
|
||||
ptSlicing = vSlicingVal[nLayCnt]
|
||||
else
|
||||
-- nel caso standard la direzione è è l'unico elemento del vettore direzioni ( visto che è comune a tutti i piani) mentre i valori sono le distanze dai piani da quello
|
||||
-- posto nell'origine
|
||||
vtSlicing = vSlicingDir[1]
|
||||
ptSlicing = ORIG() + ( vSlicingVal[nLayCnt] - dDeltaZ) * vtSlicing
|
||||
end
|
||||
EgtSetInfo( nLayId, KEY_SLICE_POS, ptSlicing)
|
||||
EgtSetInfo( nLayId, KEY_SLICE_DIR, vtSlicing)
|
||||
|
||||
-- verifico se necessario ricalcolo
|
||||
local nId = EgtGetLastInGroup( nLayId)
|
||||
local bRecalc = not nId
|
||||
@@ -820,9 +1041,10 @@ local function SlicingWithSolid( nStmId, vZSlices, dDeltaZStart, dZmin, frSlicin
|
||||
if bRecalc then
|
||||
nRecalc = nRecalc + 1
|
||||
EgtEmptyGroup( nLayId)
|
||||
dDeltaZ = dDeltaZ + EgtIf( vtRecalc and vtRecalc:getZ() > 0, -0.01, 0.01)
|
||||
dDeltaZ = dDeltaZ + EgtIf( vtRecalc and vtRecalc:getZ() > 0, - DELTAZ, DELTAZ)
|
||||
-- eseguo il ricalcolo solo a quella quota
|
||||
EgtPlaneSurfTmInters( ORIG() + ( dPosZ + dDeltaZ) * vtSlicing, vtSlicing, nStmId, nLayId, GDB_RT.GLOB, TOLER)
|
||||
local ptOn = ptSlicing + dDeltaZ * vtSlicing
|
||||
EgtPlaneSurfTmInters( ptOn, vtSlicing, nStmId, nLayId, GDB_RT.GLOB, TOLER)
|
||||
EgtOutLog( 'Warning : recalc at layer '.. EgtNumToString( nLayCnt))
|
||||
end
|
||||
EgtSetInfo( nLayId, KEY_SLICE_DELTAZ, dDeltaZ)
|
||||
@@ -910,13 +1132,10 @@ local function SlicingWithSolid( nStmId, vZSlices, dDeltaZStart, dZmin, frSlicin
|
||||
EgtSetColor( vOpenId or {}, 'RED')
|
||||
|
||||
-- creo flat region a partire dalle curve chiuse ottenute con lo slicing
|
||||
local nGrp = EgtGroup( nLayId, frSlicing, GDB_RT.GLOB)
|
||||
for i = 1, #vClosedId do
|
||||
EgtRelocateGlob( vClosedId[i], nGrp)
|
||||
EgtModifyCurveExtrusion( vClosedId[i], vtSlicing, GDB_RT.GLOB)
|
||||
EgtApproxCurve( vClosedId[i], GDB_CA.ARCS, MID_TOLER)
|
||||
EgtRelocateGlob( vClosedId[i], nLayId)
|
||||
end
|
||||
EgtErase( nGrp)
|
||||
local nSurfFR, nSrfNbr = EgtSurfFlatRegion( nLayId, vClosedId)
|
||||
|
||||
if nSurfFR then
|
||||
@@ -1056,32 +1275,69 @@ local function SlicingWithSolid( nStmId, vZSlices, dDeltaZStart, dZmin, frSlicin
|
||||
-- costolature
|
||||
local nRibsLay = EgtGetFirstNameInGroup( s_nPartId, LAY_RIBS)
|
||||
ExtractRibsLoops( nRibsLay, nStmId)
|
||||
SlicingExtraObjects( nRibsLay, vtSlicing, TYPE.RIB, RIBS_CRV, RIBS_GRP, nStmId)
|
||||
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + ( #vZSlices + 3) / nCounterTot * 100, 0) == 1 then return false end
|
||||
SlicingExtraObjects( nRibsLay, TYPE.RIB, RIBS_CRV, RIBS_GRP, nStmId)
|
||||
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + ( nCnt + 3) / nCounterTot * 100, 0) == 1 then return false end
|
||||
|
||||
local bSpiralVase = EgtGetInfo( s_nPartId, KEY_SPIRAL_VASE, 'b') or false
|
||||
if not bSpiralVase then
|
||||
|
||||
-- infill
|
||||
PrepareInfill( nStmId, vtSlicing)
|
||||
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + ( #vZSlices + 6) / nCounterTot * 100, 0) == 1 then return false end
|
||||
if nSlicingType ~= SLICING_TYPE.MULTIPLANAR then
|
||||
PrepareInfill( nStmId, vSlicingDir[1])
|
||||
end
|
||||
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + ( nCnt + 6) / nCounterTot * 100, 0) == 1 then return false end
|
||||
|
||||
-- solidi per regioni con diverso numero di passate
|
||||
local nShellNbrLay = EgtGetFirstNameInGroup( s_nPartId, LAY_SHELL_NBR)
|
||||
SlicingExtraObjects( nShellNbrLay, vtSlicing, TYPE.EXTRA_SHELL, SHELL_NBR_CRV, SHELL_NBR_GRP)
|
||||
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + ( #vZSlices + 9) / nCounterTot * 100, 0) == 1 then return false end
|
||||
SlicingExtraObjects( nShellNbrLay, TYPE.EXTRA_SHELL, SHELL_NBR_CRV, SHELL_NBR_GRP)
|
||||
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + ( nCnt + 9) / nCounterTot * 100, 0) == 1 then return false end
|
||||
|
||||
-- solidi ausiliari
|
||||
local nAuxSolidsLay = EgtGetFirstNameInGroup( s_nPartId, LAY_AUX_SOLIDS)
|
||||
SlicingExtraObjects( nAuxSolidsLay, vtSlicing, TYPE.AUX_SOLID, AUX_SOLIDS_CRV, AUX_SOLIDS_GRP)
|
||||
PrepareAuxSolidsInfill( nAuxSolidsLay, vtSlicing)
|
||||
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + ( #vZSlices + 12) / nCounterTot * 100, 0) == 1 then return false end
|
||||
|
||||
SlicingExtraObjects( nAuxSolidsLay, TYPE.AUX_SOLID, AUX_SOLIDS_CRV, AUX_SOLIDS_GRP)
|
||||
if nSlicingType ~= SLICING_TYPE.MULTIPLANAR then
|
||||
PrepareAuxSolidsInfill( nAuxSolidsLay, vSlicingDir[1])
|
||||
end
|
||||
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + ( nCnt + 12) / nCounterTot * 100, 0) == 1 then return false end
|
||||
end
|
||||
|
||||
EgtSetInfo( s_nPartId, KEY_FIRST_SOLID_LAY, nFirstSolidLay)
|
||||
EgtSetInfo( s_nPartId, KEY_LAST_SOLID_LAY, nLastSolidLay)
|
||||
|
||||
-- verifico se posso eliminare gli ultimi layer senza solido. Per poterlo fare non devono contenere setti esterni o unbounded
|
||||
local vLayers = EgtGetNameInGroup( s_nPartId, SLICE_LAYER .. '*')
|
||||
if #vLayers ~= nLastSolidLay then
|
||||
-- recupero l'ultimo layer con setti esterni
|
||||
local nLastLayerIdx
|
||||
for i = #vLayers, nLastSolidLay + 1, - 1 do
|
||||
if nLastLayerIdx then break end
|
||||
local nRibsGrp = EgtGetFirstNameInGroup( vLayers[i], RIBS_GRP)
|
||||
if nRibsGrp then
|
||||
local vRibs = EgtGetNameInGroup( nRibsGrp, RIBS_CRV .. '*')
|
||||
for j = 1, #vRibs do
|
||||
local nRibType = EgtGetInfo( vRibs[j], KEY_RIBS_TYPE, 'i')
|
||||
if nRibType == RIB_TYPE.EXTERNAL or nRibType == RIB_TYPE.UNBOUNDED then
|
||||
nLastLayerIdx = i
|
||||
break
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
if not nLastLayerIdx then
|
||||
nLastLayerIdx = nLastSolidLay
|
||||
end
|
||||
-- cancello eventuali layers inutile
|
||||
for i = #vLayers, nLastLayerIdx + 1, -1 do
|
||||
EgtErase( vLayers[i])
|
||||
end
|
||||
end
|
||||
|
||||
-- se multiplanare verifico strand massimo e minimo
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
if not VerifyMultiPlanarStrand() then
|
||||
return false
|
||||
end
|
||||
end
|
||||
|
||||
return true
|
||||
end
|
||||
|
||||
@@ -1105,10 +1361,26 @@ function CalcSlices.Exec( nPartId, nStmId)
|
||||
vtSlicing = VectorFromSpherical( 1, 45, dHorAng)
|
||||
elseif nSlicingType == SLICING_TYPE.HORIZONTAL then
|
||||
vtSlicing = X_AX()
|
||||
elseif nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
-- la direzione può cambiare da un piano all'altro
|
||||
vtSlicing = Z_AX()
|
||||
end
|
||||
|
||||
EgtSetInfo( s_nPartId, KEY_SLICING_DIR, vtSlicing)
|
||||
local frSlicing = Frame3d( ORIG(), vtSlicing)
|
||||
EgtSetInfo( s_nPartId, KEY_SLICE_DIR, vtSlicing)
|
||||
-- per compatibilità con vecchie macchine
|
||||
EgtSetInfo( s_nPartId, 'SlicingDir', vtSlicing)
|
||||
|
||||
-- recupero parametri di slicing
|
||||
local dSliceStep = EgtGetInfo( s_nPartId, KEY_SLICE_STEP, 'd') or 10000
|
||||
if dSliceStep < GEO.EPS_SMALL then
|
||||
EgtOutLog( 'Warning : SliceStep is 0')
|
||||
return
|
||||
end
|
||||
s_dStrand = EgtGetInfo( s_nPartId, KEY_STRAND, 'd')
|
||||
local nMaxSlicesNbr = EgtGetInfo( s_nPartId, KEY_MAX_SLICES_NBR, 'i') or 0
|
||||
if nMaxSlicesNbr == 0 then
|
||||
nMaxSlicesNbr = GEO.INFINITO -- da gestire come caso illimitato
|
||||
end
|
||||
|
||||
-- recupero la superficie ed eventuali setti esterni/unbounded da usare come riferimenti per quote slicing
|
||||
local vRefIds = {}
|
||||
@@ -1116,6 +1388,10 @@ function CalcSlices.Exec( nPartId, nStmId)
|
||||
local nRibsLay = EgtGetFirstNameInGroup( s_nPartId, LAY_RIBS)
|
||||
local vRibsIds = EgtGetAllInGroup( nRibsLay)
|
||||
local bLimitUnbddRibs = EgtGetInfo( s_nPartId, KEY_LIMIT_UNBDD_RIBS, 'b') or false
|
||||
if bLimitUnbddRibs and not nStmId then
|
||||
EgtOutBox( "No solid to limit unbounded ribs!", 'Error', 'ERROR')
|
||||
return
|
||||
end
|
||||
for i = 1, #vRibsIds do
|
||||
if EgtGetType( vRibsIds[i]) == GDB_TY.SRF_MESH then
|
||||
local nType = ReadParam( vRibsIds[i], KEY_RIBS_TYPE, 'i', RIB_TYPE.INTERNAL)
|
||||
@@ -1124,58 +1400,40 @@ function CalcSlices.Exec( nPartId, nStmId)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
if bLimitUnbddRibs and not nStmId then
|
||||
EgtOutBox( "No solid to limit unbounded ribs!", 'Error', 'ERROR')
|
||||
return
|
||||
end
|
||||
|
||||
-- recupero il box
|
||||
local b3Box = BBox3d()
|
||||
for i = 1, #vRefIds do
|
||||
local b3Tmp = EgtGetBBoxRef( vRefIds[i], GDB_BB.STANDARD, frSlicing)
|
||||
b3Box:Add( b3Tmp)
|
||||
end
|
||||
if not b3Box or b3Box:isEmpty() then
|
||||
if #vRefIds == 0 then
|
||||
EgtOutBox( "No part to be sliced!", 'Error', 'ERROR')
|
||||
return
|
||||
return false
|
||||
end
|
||||
|
||||
-- Parametri di slicing
|
||||
local dSliceStep = EgtGetInfo( s_nPartId, KEY_SLICE_STEP, 'd') or 10000
|
||||
if dSliceStep < GEO.EPS_SMALL then
|
||||
EgtOutLog( 'Warning : SliceStep is 0')
|
||||
return
|
||||
end
|
||||
s_dStrand = EgtGetInfo( s_nPartId, KEY_STRAND, 'd')
|
||||
|
||||
local dZmin = b3Box:getMin():getZ()
|
||||
local dZmax = b3Box:getMax():getZ()
|
||||
local dMaxH = ComputeMaxH( vRefIds, frSlicing, dSliceStep)
|
||||
dZmax = min( dZmax, dMaxH)
|
||||
-- calcolo i dati dello slicing
|
||||
local dMaxH = ComputeMaxH( vRefIds, vtSlicing, dSliceStep)
|
||||
local vSlicingVal = {}
|
||||
local vSlicingDir = {}
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
vSlicingVal, vSlicingDir = ComputeMultiplanarSlicingData( dSliceStep, nMaxSlicesNbr)
|
||||
else
|
||||
vSlicingVal, vSlicingDir = ComputeSlicingData( vRefIds, vtSlicing, dSliceStep, dMaxH, nMaxSlicesNbr)
|
||||
end
|
||||
if not vSlicingVal then
|
||||
return false
|
||||
end
|
||||
|
||||
-- Eseguo slicing
|
||||
local nLayCnt = 1
|
||||
local nRecalc = 0
|
||||
local dDeltaZStart = 0.2
|
||||
|
||||
local vZSlices = ComputeZSlices( dSliceStep, dZmin, dDeltaZStart, dZmax)
|
||||
|
||||
local bOk = true
|
||||
if not nStmId then
|
||||
-- caso senza solido e solo setti
|
||||
bOk = SlicingNoSolid( nRibsLay, vZSlices, dDeltaZStart, dZmin, frSlicing)
|
||||
bOk = SlicingNoSolid( nRibsLay, nSlicingType, dMaxH, dSliceStep, vSlicingVal, vSlicingDir)
|
||||
else
|
||||
-- caso con solido
|
||||
bOk = SlicingWithSolid( nStmId, vZSlices, dDeltaZStart, dZmin, frSlicing)
|
||||
bOk = SlicingWithSolid( nStmId, nSlicingType, dMaxH, vSlicingVal, vSlicingDir)
|
||||
end
|
||||
|
||||
-- eventuale segnalazione errori
|
||||
if bOk and #s_vErr > 0 then
|
||||
EgtOutBox( 'Slicing Error on layers :\n' .. table.concat( s_vErr, ','), 'SlicingCalc')
|
||||
end
|
||||
end
|
||||
|
||||
return bOk
|
||||
return bOk
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
|
||||
+583
-80
@@ -1,4 +1,4 @@
|
||||
-- CalcToolPath.lua by Egaltech s.r.l. 2022/12/20
|
||||
-- CalcToolPath.lua by Egaltech s.r.l. 2026/02/16
|
||||
-- Calcolo percorsi di lavoro per Stampa 3d
|
||||
|
||||
-- Tabella per definizione modulo
|
||||
@@ -25,6 +25,7 @@ local s_dSpralVaseFirstDelta = 30 -- lunghezza del tratto sul primo layer che
|
||||
local function GetLayerParamsForToolPathCalc()
|
||||
local LayerParams = {}
|
||||
LayerParams.bSpiralVase = EgtGetInfo( s_nPartId, KEY_SPIRAL_VASE, 'b') or false
|
||||
LayerParams.dSpiralVaseLen = EgtGetInfo( s_nPartId, KEY_SPIRAL_VASE_LEN, 'd') or 0
|
||||
LayerParams.dStrand = EgtGetInfo( s_nPartId, KEY_STRAND, 'd')
|
||||
LayerParams.dInnerStrand = EgtGetInfo( s_nPartId, KEY_INNER_STRAND, 'd') or LayerParams.dStrand
|
||||
if LayerParams.dInnerStrand < GEO.EPS_SMALL then LayerParams.dInnerStrand = LayerParams.dStrand end
|
||||
@@ -46,7 +47,7 @@ local function GetLayerParamsForToolPathCalc()
|
||||
LayerParams.dCoastingLen = EgtGetInfo( s_nPartId, KEY_COASTING_LEN, 'd')
|
||||
LayerParams.dWipeLen = EgtGetInfo( s_nPartId, KEY_WIPE_LEN, 'd')
|
||||
LayerParams.dWipeDir = EgtGetInfo( s_nPartId, KEY_WIPE_DIR, 'd') or 0.0
|
||||
LayerParams.vtSlicing = EgtGetInfo( s_nPartId, KEY_SLICING_DIR, 'v')
|
||||
LayerParams.vtSlicing = EgtGetInfo( s_nPartId, KEY_SLICE_DIR, 'v')
|
||||
LayerParams.dTDiam = EgtGetInfo( s_nPartId, KEY_TOOL_DIAM, 'd')
|
||||
-- parametri infill
|
||||
LayerParams.dInfillStrand = EgtGetInfo( s_nPartId, KEY_INFILL_STRAND, 'd') or LayerParams.dStrand
|
||||
@@ -64,56 +65,103 @@ local function GetLayerParamsForToolPathCalc()
|
||||
-- parametri da file ini del programma
|
||||
local sIniFile = EgtGetIniFile()
|
||||
LayerParams.dSpiralVaseInterpLen = EgtGetNumberFromIni( '3dPrinting', KEY_SPIRAL_VASE_INTERP_LEN, 100.0, sIniFile)
|
||||
LayerParams.dSpiralVaseLen = EgtGetNumberFromIni( '3dPrinting', KEY_SPIRAL_VASE_LEN, 0.0, sIniFile)
|
||||
|
||||
return LayerParams
|
||||
end
|
||||
|
||||
------------------------------------------------------------------
|
||||
local function ComputeToolPathBox( nTpathGrpId)
|
||||
|
||||
-- calcolo il box del toolpath (escludendo i wipe)
|
||||
local function ComputeToolPathBox( nLayerId)
|
||||
-- calcolo il box di tutti i toolpath del layer escludendo i wipe
|
||||
local b3Box = BBox3d()
|
||||
local nEntId = EgtGetFirstInGroup( nTpathGrpId)
|
||||
while nEntId do
|
||||
if EgtGetInfo( nEntId, KEY_TYPE, 'i') ~= TYPE.WIPE then
|
||||
local b3Ent = EgtGetBBoxGlob( nEntId, GDB_BB.STANDARD)
|
||||
b3Box:Add( b3Ent)
|
||||
local vCrvGrpIds = EgtGetNameInGroup( nLayerId, CONTOUR_GRP .. '*')
|
||||
for i = 1, #vCrvGrpIds do
|
||||
-- recupero tutti gli elementi del toolpath
|
||||
local nTPathGrpId = EgtGetFirstNameInGroup( vCrvGrpIds[i], TOOLPATH_GRP)
|
||||
local nEntId = EgtGetFirstInGroup( nTPathGrpId)
|
||||
while nEntId do
|
||||
if EgtGetInfo( nEntId, KEY_TYPE, 'i') ~= TYPE.WIPE then
|
||||
local b3Ent = EgtGetBBoxGlob( nEntId, GDB_BB.STANDARD)
|
||||
b3Box:Add( b3Ent)
|
||||
end
|
||||
nEntId = EgtGetNext( nEntId)
|
||||
end
|
||||
nEntId = EgtGetNext( nEntId)
|
||||
end
|
||||
|
||||
return b3Box
|
||||
end
|
||||
|
||||
------------------------------------------------------------------
|
||||
local function AddZCorrection( b3Box, LayerParams)
|
||||
local function AddZCorrection( LayerParams)
|
||||
|
||||
local dNxy = sqrt( LayerParams.vtSlicing:getX() * LayerParams.vtSlicing:getX() + LayerParams.vtSlicing:getY() * LayerParams.vtSlicing:getY())
|
||||
local dNz = LayerParams.vtSlicing:getZ()
|
||||
-- altezza necessaria per lo strand
|
||||
local dHStrand = 0.5 * LayerParams.dStrand * dNxy + LayerParams.dLayHeight * dNz
|
||||
-- altezza necessaria per il tool
|
||||
local dHTool = LayerParams.dTDiam / 2 * dNxy
|
||||
-- altezza disponibile
|
||||
local dHBox = EgtGetInfo( s_nPartId, KEY_BOX_MIN_Z, 'i') or 0
|
||||
local dHDisp = b3Box:getMin():getZ() - max( dHBox, 0)
|
||||
|
||||
local dCorrZ = max( 0, max( dHStrand, dHTool) - dHDisp)
|
||||
|
||||
-- applico la correzione al pezzo
|
||||
local vtMove = dCorrZ * Z_AX()
|
||||
EgtMove( s_nPartId, vtMove, GDB_RT.GLOB)
|
||||
EgtSetInfo( s_nPartId, KEY_MOVED_PART2, vtMove)
|
||||
-- correggo la posizione del riferimento
|
||||
local nFrameId = EgtGetFirstNameInGroup( s_nPartId, LAY_FRAME)
|
||||
EgtMove( nFrameId, - vtMove, GDB_RT.GLOB)
|
||||
|
||||
-- correzione per i wipe
|
||||
-- altezza minima necessaria per i wipe
|
||||
local dHMinWipe = LayerParams.dTDiam / 2 * dNxy + max( dHBox, 0) + s_dHSafeWipe
|
||||
local vLayIds = EgtGetNameInGroup( s_nPartId, SLICE_LAYER.."*")
|
||||
local nSlicingType = EgtGetInfo( s_nPartId, KEY_SLICING_TYPE, 'i')
|
||||
|
||||
-- 1) correzione per il pezzo
|
||||
local dHBox = EgtGetInfo( s_nPartId, KEY_BOX_MIN_Z, 'i') or 0
|
||||
local dCorrZ = 0
|
||||
|
||||
-- se slicing multiplanare vtSlicing cambia ad ogni layer quindi la correzione deve essere calcolata sul singolo layer
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
for i = 1, #vLayIds do
|
||||
-- calcolo i valori legati a vtSlicing per il layer corrente
|
||||
local vtSlicing = EgtGetInfo( vLayIds[i], KEY_SLICE_DIR, 'v')
|
||||
local dNxy = sqrt( vtSlicing:getX() * vtSlicing:getX() + vtSlicing:getY() * vtSlicing:getY())
|
||||
local dNz = vtSlicing:getZ()
|
||||
-- altezza necessaria per lo strand non serve perchè l'altezza viene adattata lungo il percorso
|
||||
-- altezza necessaria per il tool
|
||||
local dHTool = LayerParams.dTDiam / 2 * dNxy
|
||||
-- altezza disponibile
|
||||
local b3Layer = ComputeToolPathBox( vLayIds[i])
|
||||
local dHDisp = b3Layer:getMin():getZ() - max( dHBox, 0)
|
||||
|
||||
local dCurrCorrZ = max( 0, dHTool - dHDisp)
|
||||
dCorrZ = max( dCorrZ, dCurrCorrZ)
|
||||
end
|
||||
|
||||
-- se slicing standard la correzione può essere calcolata sull'intero pezzo
|
||||
else
|
||||
local dNxy = sqrt( LayerParams.vtSlicing:getX() * LayerParams.vtSlicing:getX() + LayerParams.vtSlicing:getY() * LayerParams.vtSlicing:getY())
|
||||
local dNz = LayerParams.vtSlicing:getZ()
|
||||
-- altezza necessaria per lo strand
|
||||
local dHStrand = 0.5 * LayerParams.dStrand * dNxy + LayerParams.dLayHeight * dNz
|
||||
-- altezza necessaria per il tool
|
||||
local dHTool = LayerParams.dTDiam / 2 * dNxy
|
||||
-- altezza disponibile
|
||||
-- calcolo il box complessivo dei percorsi
|
||||
local b3Tot = BBox3d()
|
||||
for i = 1, #vLayIds do
|
||||
local b3Curr = ComputeToolPathBox( vLayIds[i])
|
||||
b3Tot:Add( b3Curr)
|
||||
end
|
||||
local dHDisp = b3Tot:getMin():getZ() - max( dHBox, 0)
|
||||
|
||||
dCorrZ = max( 0, max( dHStrand, dHTool) - dHDisp)
|
||||
end
|
||||
|
||||
-- applico la correzione al pezzo
|
||||
if dCorrZ > GEO.EPS_SMALL then
|
||||
local vtMove = dCorrZ * Z_AX()
|
||||
EgtMove( s_nPartId, vtMove, GDB_RT.GLOB)
|
||||
EgtSetInfo( s_nPartId, KEY_MOVED_PART2, vtMove)
|
||||
-- correggo la posizione del riferimento
|
||||
local nFrameId = EgtGetFirstNameInGroup( s_nPartId, LAY_FRAME)
|
||||
EgtMove( nFrameId, - vtMove, GDB_RT.GLOB)
|
||||
end
|
||||
|
||||
-- 2) correzione per i wipe
|
||||
-- altezza minima necessaria per i wipe. Se slicing multiplanare l'altezza minima per il wipe cambia ad ogni layer, altrimenti è un valore comune
|
||||
local dHMinWipe
|
||||
if nSlicingType ~= SLICING_TYPE.MULTIPLANAR then
|
||||
local dNxy = sqrt( LayerParams.vtSlicing:getX() * LayerParams.vtSlicing:getX() + LayerParams.vtSlicing:getY() * LayerParams.vtSlicing:getY())
|
||||
dHMinWipe = LayerParams.dTDiam / 2 * dNxy + max( dHBox, 0) + s_dHSafeWipe
|
||||
end
|
||||
|
||||
for i = 1, #vLayIds do
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
local vtSlicing = EgtGetInfo( vLayIds[i], KEY_SLICE_DIR, 'v')
|
||||
local dNxy = sqrt( vtSlicing:getX() * vtSlicing:getX() + vtSlicing:getY() * vtSlicing:getY())
|
||||
dHMinWipe = LayerParams.dTDiam / 2 * dNxy + max( dHBox, 0) + s_dHSafeWipe
|
||||
end
|
||||
|
||||
local vCrvGrps = EgtGetNameInGroup( vLayIds[i], CONTOUR_GRP.."*")
|
||||
for j = 1, #vCrvGrps do
|
||||
local nTPath = EgtGetFirstNameInGroup( vCrvGrps[j], TOOLPATH_GRP)
|
||||
@@ -364,6 +412,7 @@ end
|
||||
--------------------------------------------------------------------
|
||||
local function AddLeadOut( nCrvId, LayerParams, nGrpId)
|
||||
|
||||
local vtSlicing = EgtCurveExtrusion( nCrvId, GDB_ID.ROOT)
|
||||
local ptS = EgtEP( nCrvId, GDB_ID.ROOT)
|
||||
local vtTang = EgtEV( nCrvId, GDB_ID.ROOT)
|
||||
local vtOrtho = Vector3d( vtTang)
|
||||
@@ -373,7 +422,7 @@ local function AddLeadOut( nCrvId, LayerParams, nGrpId)
|
||||
dAng = 90
|
||||
end
|
||||
|
||||
vtOrtho:rotate( LayerParams.vtSlicing, dAng)
|
||||
vtOrtho:rotate( vtSlicing, dAng)
|
||||
local ptE = ptS + LayerParams.dLeadOutTangDist * vtTang + LayerParams.dLeadOutOrthoDist * vtOrtho
|
||||
|
||||
local nLeadOutCrv
|
||||
@@ -385,7 +434,7 @@ local function AddLeadOut( nCrvId, LayerParams, nGrpId)
|
||||
|
||||
if nLeadOutCrv then
|
||||
EgtRelocate( nLeadOutCrv, nCrvId, GDB_IN.AFTER)
|
||||
EgtModifyCurveExtrusion( nLeadOutCrv, LayerParams.vtSlicing, GDB_RT.GLOB)
|
||||
EgtModifyCurveExtrusion( nLeadOutCrv, vtSlicing, GDB_RT.GLOB)
|
||||
EgtSetInfo( nLeadOutCrv, KEY_TYPE, TYPE.LINK)
|
||||
EgtSetName( nLeadOutCrv, LEAD_OUT_CRV)
|
||||
EgtSetColor( nLeadOutCrv, EgtStdColor('GRAY'))
|
||||
@@ -396,9 +445,10 @@ local function AddLeadOut( nCrvId, LayerParams, nGrpId)
|
||||
end
|
||||
|
||||
--------------------------------------------------------------------
|
||||
local function AddRetraction( nCrvId, vtSlicing, dCoastingLen, dWipeLen, dWipeDir)
|
||||
local function AddRetraction( nCrvId, dCoastingLen, dWipeLen, dWipeDir)
|
||||
|
||||
-- recupero i parametri per retrazione
|
||||
local vtSlicing = EgtCurveExtrusion( nCrvId, GDB_ID.ROOT)
|
||||
local nType = EgtGetInfo( nCrvId, KEY_TYPE, 'i')
|
||||
local bClosed = EgtGetInfo( nCrvId, KEY_CLOSED_CRV, 'b') or false
|
||||
local bInverted = EgtGetInfo( nCrvId, KEY_INVERTED_CRV, 'b') or false
|
||||
@@ -529,22 +579,22 @@ end
|
||||
local function AddRetractionOnLastCrv( nCrvId, nTpathGrpId, LayerParams, dCoastingLen, dWipeLen, dWipeDir)
|
||||
|
||||
if LayerParams.nLeadOutType == LEAD_TYPE.NONE then
|
||||
AddRetraction( nCrvId, LayerParams.vtSlicing, dCoastingLen, dWipeLen, dWipeDir)
|
||||
AddRetraction( nCrvId, dCoastingLen, dWipeLen, dWipeDir)
|
||||
else
|
||||
local nLeadOutId = EgtGetLastInGroup( nTpathGrpId)
|
||||
local dLen = EgtCurveLength( nLeadOutId)
|
||||
if dLen > dCoastingLen - 500 * GEO.EPS_SMALL then
|
||||
-- coinvolge solo la curva di lead out
|
||||
local dNewCoastingLen = EgtIf( abs( dCoastingLen - dLen) < 500 * GEO.EPS_SMALL, dLen, dCoastingLen) -- verifico se interamente coinvolta
|
||||
AddRetraction( nLeadOutId, LayerParams.vtSlicing, dNewCoastingLen, dWipeLen, dWipeDir)
|
||||
AddRetraction( nLeadOutId, dNewCoastingLen, dWipeLen, dWipeDir)
|
||||
else
|
||||
-- coinvolge parte dell'ultima shell crv
|
||||
local dNewCoastingLen = dCoastingLen - dLen
|
||||
local nCoastingId = AddRetraction( nCrvId, LayerParams.vtSlicing, dNewCoastingLen, 0.0, dWipeDir)
|
||||
local nCoastingId = AddRetraction( nCrvId, dNewCoastingLen, 0.0, dWipeDir)
|
||||
EgtAddCurveCompoCurve( nCoastingId, nLeadOutId)
|
||||
EgtSetInfo( nCoastingId, KEY_CLOSED_CRV, 0)
|
||||
-- wipe
|
||||
AddRetraction( nCoastingId, LayerParams.vtSlicing, 0, dWipeLen, dWipeDir)
|
||||
AddRetraction( nCoastingId, 0, dWipeLen, dWipeDir)
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -589,7 +639,7 @@ local function CalcShellsToolPath( vEntIds, nTpathGrpId, LayerParams)
|
||||
for i = 1, #vIds - 1 do
|
||||
local nNextId = EgtGetNext( vIds[i])
|
||||
if EgtGetInfo( nNextId, KEY_TYPE, 'i') ~= TYPE.LINK then
|
||||
AddRetraction( vIds[i], LayerParams.vtSlicing, LayerParams.dCoastingLen, LayerParams.dWipeLen, LayerParams.dWipeDir)
|
||||
AddRetraction( vIds[i], LayerParams.dCoastingLen, LayerParams.dWipeLen, LayerParams.dWipeDir)
|
||||
end
|
||||
end
|
||||
-- sull'ultima curva gestione speciale per eventuale lead out
|
||||
@@ -673,7 +723,7 @@ local function CalcExtraShellToolPath( vEntIds, nTpathGrpId, LayerParams)
|
||||
dWipeDir = LayerParams.dWipeDir
|
||||
end
|
||||
|
||||
AddRetraction( vIds[i], LayerParams.vtSlicing, dCoastingLen, dWipeLen, dWipeDir)
|
||||
AddRetraction( vIds[i], dCoastingLen, dWipeLen, dWipeDir)
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -900,7 +950,7 @@ local function CalcInfillToolPath( nInfillGrp, nTpathGrpId, LayerParams)
|
||||
for i = 1, #vIds do
|
||||
local nNext = EgtGetNext( vIds[i])
|
||||
if not nNext or EgtGetName( nNext) ~= LINK_CRV then
|
||||
AddRetraction( vIds[i], LayerParams.vtSlicing, LayerParams.dInfillCoasting, LayerParams.dInfillWipe, LayerParams.dInfillWipeDir)
|
||||
AddRetraction( vIds[i], LayerParams.dInfillCoasting, LayerParams.dInfillWipe, LayerParams.dInfillWipeDir)
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -943,7 +993,7 @@ local function CalcSolidFillToolPath( nInfillGrp, nTpathGrpId, LayerParams)
|
||||
for i = 1, #vIds do
|
||||
local nNextId = EgtGetNext( vIds[i])
|
||||
if not nNextId or EgtGetInfo( nNextId, KEY_TYPE, 'i') ~= TYPE.LINK then
|
||||
AddRetraction( vIds[i], LayerParams.vtSlicing, LayerParams.dInfillCoasting, LayerParams.dInfillWipe, LayerParams.dInfillWipeDir)
|
||||
AddRetraction( vIds[i], LayerParams.dInfillCoasting, LayerParams.dInfillWipe, LayerParams.dInfillWipeDir)
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -1066,7 +1116,7 @@ local function CalcAuxSolidsToolPath( nAuxSolidsGrp, nAuxSolidsPathGrp, nTpathGr
|
||||
for i = 1, #vIds do
|
||||
local nNextId = EgtGetNext( vIds[i])
|
||||
if not nNextId or EgtGetName( nNextId) ~= LINK_CRV then
|
||||
AddRetraction( vIds[i], LayerParams.vtSlicing, dCoastingLen, dWipeLen, dWipeDir)
|
||||
AddRetraction( vIds[i], dCoastingLen, dWipeLen, dWipeDir)
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -1752,7 +1802,7 @@ local function AddRibsLeadOut( nCrv, nLoopsGrp, vtSlicing, nGrpTmp, bForceNoSoli
|
||||
|
||||
-- se setto chiuso applico la stessa uscita delle shell
|
||||
if EgtCurveIsClosed( nCrv) then
|
||||
return AddRetraction( nCrv, vtSlicing, dRibsLOCoasting, dRibsLOWipe, dRibsLOWipeAng)
|
||||
return AddRetraction( nCrv, dRibsLOCoasting, dRibsLOWipe, dRibsLOWipeAng)
|
||||
end
|
||||
|
||||
local ptE = EgtEP( nCrv, GDB_ID.ROOT)
|
||||
@@ -1839,7 +1889,7 @@ local function AddRibsLeadOut( nCrv, nLoopsGrp, vtSlicing, nGrpTmp, bForceNoSoli
|
||||
end
|
||||
|
||||
-- se non ho trovato curva sul bordo su cui fare il lead out aggiungo coasting e wipe come sulle shell normali
|
||||
AddRetraction( nCrv, vtSlicing, dRibsLOCoasting, dRibsLOWipe, dRibsLOWipeAng)
|
||||
AddRetraction( nCrv, dRibsLOCoasting, dRibsLOWipe, dRibsLOWipeAng)
|
||||
|
||||
end
|
||||
|
||||
@@ -2010,32 +2060,299 @@ local function AddSpiralVaseLeadOut( nOldId, LayerParams)
|
||||
end
|
||||
AddRetractionOnLastCrv( nOldId, EgtGetParent( nOldId), LayerParams, LayerParams.dCoastingLen, LayerParams.dWipeLen, LayerParams.dWipeDir)
|
||||
else
|
||||
AddRetraction( nOldId, LayerParams.vtSlicing, LayerParams.dCoastingLen, LayerParams.dWipeLen, LayerParams.dWipeDir)
|
||||
AddRetraction( nOldId, LayerParams.dCoastingLen, LayerParams.dWipeLen, LayerParams.dWipeDir)
|
||||
end
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function SpiralVaseFullMultiPlanar( vLayIds, LayerParams)
|
||||
-- la differenza in altezza viene distribuita uniformemente lungo tutto il percorso. Tale differenza non è costante per tutti i punti della curva ma varia da punto a punto
|
||||
-- la continuità tra i layers viene risolta sul layer corrente ( uniformemente su tutto il tratto o solo nella parte iniziale di lunghezza dSpiralVaseInterpLen)
|
||||
-- gestione speciale per i primi due layers
|
||||
-- aggiunta passata extra finale piana alla quota dell'ultimo layer
|
||||
|
||||
local nOldId, nOldPathId
|
||||
local nRealLayer = 1
|
||||
|
||||
-- ciclo sui layer
|
||||
for nIdx = 1, #vLayIds do
|
||||
|
||||
-- recupero il piano di slicing
|
||||
local vtSlicing = EgtGetInfo( vLayIds[nIdx], KEY_SLICE_DIR, 'v')
|
||||
local ptSlicing = EgtGetInfo( vLayIds[nIdx], KEY_SLICE_POS, 'p')
|
||||
|
||||
-- cerco i gruppi di contorni
|
||||
local vCrvGrpIds = EgtGetNameInGroup( vLayIds[ nIdx], CONTOUR_GRP.."*")
|
||||
if #vCrvGrpIds > 1 then
|
||||
-- se più di un gruppo di curve errore
|
||||
EgtOutBox( 'Error in spiral vase : layer ' .. tostring( nIdx) .. ' has more than one toolpath', 'ToolPathCalc')
|
||||
return false
|
||||
end
|
||||
|
||||
-- recupero il gruppo dei percorsi
|
||||
local nPathGrpId = EgtGetFirstNameInGroup( vCrvGrpIds[1], PATH_GRP)
|
||||
if not nPathGrpId then
|
||||
EgtOutBox( 'Error missing paths', 'ToolPathCalc')
|
||||
return
|
||||
else
|
||||
EgtSetStatus( nPathGrpId, GDB_ST.OFF)
|
||||
end
|
||||
|
||||
-- recupero il gruppo dei percorsi utensile
|
||||
local nTpathGrpId = EgtGetFirstNameInGroup( vCrvGrpIds[1], TOOLPATH_GRP)
|
||||
if not nTpathGrpId then
|
||||
nTpathGrpId = EgtGroup( vCrvGrpIds[1])
|
||||
EgtSetName( nTpathGrpId, TOOLPATH_GRP)
|
||||
else
|
||||
EgtEmptyGroup( nTpathGrpId)
|
||||
end
|
||||
|
||||
-- creo il percorso di lavoro :
|
||||
local vEntIds = EgtGetAllInGroup( nPathGrpId)
|
||||
if #vEntIds > 1 then
|
||||
EgtOutBox( 'Error in spiral vase : layer ' .. tostring( nIdx) .. ' has more than one toolpath', 'ToolPathCalc')
|
||||
return false
|
||||
end
|
||||
|
||||
local nNewEntId = EgtCopyGlob( vEntIds[1] or GDB_ID.NULL, nTpathGrpId, GDB_IN.LAST_SON)
|
||||
if nNewEntId then
|
||||
|
||||
EgtModifyCurveExtrusion( nNewEntId, vtSlicing, GDB_RT.GLOB)
|
||||
EgtSetInfo( nNewEntId, KEY_CRV_STRAND, LayerParams.dStrand)
|
||||
-- eventuale inversione
|
||||
if LayerParams.bInvert then
|
||||
EgtInvertCurve( nNewEntId)
|
||||
EgtSetInfo( nNewEntId, KEY_INVERTED_CRV, 1)
|
||||
end
|
||||
EgtSetColor( nNewEntId, EgtStdColor('GRAY'))
|
||||
|
||||
-- se primo layer
|
||||
if nRealLayer == 1 then
|
||||
-- mi sposto dell'altezza layer
|
||||
EgtMove( nNewEntId, LayerParams.dLayHeight * vtSlicing, GDB_RT.GLOB)
|
||||
|
||||
-- eventuale lead in
|
||||
if LayerParams.nLeadInType ~= LEAD_TYPE.NONE then
|
||||
EgtTrimCurveStartAtLen( nNewEntId, LayerParams.dOffsetLP)
|
||||
local nLeadInCrv = AddLeadIn( nNewEntId, LayerParams, nTpathGrpId)
|
||||
if nLeadInCrv and LayerParams.bLinearApprox then
|
||||
EgtApproxCurve( nLeadInCrv, GDB_CA.LINES, LayerParams.dLinearApproxTol)
|
||||
end
|
||||
end
|
||||
|
||||
else
|
||||
-- recupero il piano precedente
|
||||
local vtSlicingPrev = EgtGetInfo( vLayIds[nIdx-1], KEY_SLICE_DIR, 'v')
|
||||
local ptSlicingPrev = EgtGetInfo( vLayIds[nIdx-1], KEY_SLICE_POS, 'p') + LayerParams.dLayHeight * vtSlicingPrev
|
||||
local dCosAng = vtSlicing * vtSlicingPrev
|
||||
|
||||
-- a) continuità dei percorsi
|
||||
local ptEPrev = EgtEP( nOldId, GDB_ID.ROOT)
|
||||
local ptEProj = ptEPrev - ( ptEPrev - ptSlicing) * vtSlicing * vtSlicing
|
||||
|
||||
-- tento di creare la continuità spostando il punto ( caso in cui ptEProj poggia su nNewEntId)
|
||||
EgtChangeClosedCurveStartPoint( nNewEntId, ptEProj, GDB_RT.GLOB)
|
||||
if not AreSamePointApprox( ptEPrev, EgtSP( nNewEntId, GDB_ID.ROOT)) then
|
||||
-- se i percorsi non sono in continuità modifico percorso corrente usando come guida la proiezione ortogonale del percorso precedente
|
||||
-- sul piano corrente ( SpiralizeAlongGuide lavora con curve piane complanari)
|
||||
|
||||
local nProjCrv = EgtCopyGlob( nOldPathId, nPathGrpId)
|
||||
if LayerParams.bInvert then
|
||||
EgtInvertCurve( nProjCrv)
|
||||
end
|
||||
EgtMove( nProjCrv, LayerParams.dLayHeight * vtSlicingPrev, GDB_RT.GLOB)
|
||||
EgtProjectCurveOnPlane( nProjCrv, ptSlicing, vtSlicing, GDB_RT.GLOB)
|
||||
EgtChangeClosedCurveStartPoint( nProjCrv, ptEProj, GDB_RT.GLOB)
|
||||
|
||||
-- approssimo con tratti lineari imponendo lunghezza massima per migliorare spiralize
|
||||
EgtApproxCurve( nProjCrv, GDB_CA.SPECIAL_LINES, 0.01, s_dSpiralVaseMaxLen)
|
||||
EgtApproxCurve( nNewEntId, GDB_CA.SPECIAL_LINES, 0.01, s_dSpiralVaseMaxLen)
|
||||
|
||||
-- 1) la differenza va distribuita su tutta la curva
|
||||
local dLen = EgtCurveLength( nNewEntId)
|
||||
if LayerParams.dSpiralVaseInterpLen < GEO.EPS_SMALL or LayerParams.dSpiralVaseInterpLen > dLen - GEO.EPS_SMALL then
|
||||
EgtSpiralizeCurveAlongGuide( nNewEntId, nProjCrv)
|
||||
|
||||
-- 2) la differenza va distribuita solo su un sottotratto
|
||||
else
|
||||
local dParSplit = EgtCurveParamAtLength( nNewEntId, LayerParams.dSpiralVaseInterpLen)
|
||||
local nNewEndId2 = EgtSplitCurveAtParam( nNewEntId, dParSplit)
|
||||
-- trim della curva guida
|
||||
local _, _, dParMinDist = EgtPointCurveDist( EgtEP( nNewEntId, GDB_ID.ROOT), nProjCrv, GDB_RT.GLOB)
|
||||
local nGuideId2 = EgtSplitCurveAtParam( nProjCrv, dParMinDist)
|
||||
EgtErase( nGuideId2)
|
||||
-- spiralize solo del primo pezzo e riassemblo la curva
|
||||
EgtSpiralizeCurveAlongGuide( nNewEntId, nProjCrv)
|
||||
EgtAddCurveCompoCurve( nNewEntId, nNewEndId2)
|
||||
end
|
||||
-- forzo il punto
|
||||
EgtModifyCurveStartPoint( nNewEntId, ptEProj, GDB_RT.GLOB)
|
||||
EgtErase( nProjCrv)
|
||||
end
|
||||
|
||||
-- b) spostamento verticale : alzo la curva e, per creare una transizione uniforme dal layer precedente, abbasso i suoi punti in modo graduale sul valore della distanza dal
|
||||
-- dal layer precedente
|
||||
EgtMove( nNewEntId, LayerParams.dLayHeight * vtSlicing, GDB_RT.GLOB)
|
||||
|
||||
-- approssimazione lineare per migliorare il calcolo delle quote
|
||||
EgtApproxCurve( nNewEntId, GDB_CA.SPECIAL_LINES, 0.01, s_dSpiralVaseMaxLen)
|
||||
|
||||
if nRealLayer == 2 then
|
||||
-- per non collassare sul layer precedente che è piatto alzo la prima metà della curva sempre di 0.5 * Delta
|
||||
local dLen = EgtCurveLength( nNewEntId)
|
||||
local dParMid = EgtCurveParamAtLength( nNewEntId, 0.5 * dLen)
|
||||
EgtAddCurveCompoJoint( nNewEntId, dParMid)
|
||||
local vDelta = {}
|
||||
local _, dE = EgtCurveDomain( nNewEntId)
|
||||
for dU = 0, dE - 1 do
|
||||
local ptCurr = EgtUP( nNewEntId, dU, GDB_ID.ROOT)
|
||||
local dDist = ( ( ptCurr - ptSlicingPrev) * vtSlicingPrev) / dCosAng -- distanza dal piano precedente lungo vtSlicing
|
||||
local dCurrLen = EgtCurveLengthAtParam( nNewEntId, dU)
|
||||
if dCurrLen < 0.5 * dLen then
|
||||
vDelta[dU] = - dDist * 0.5
|
||||
else
|
||||
vDelta[dU] = - dDist * ( dLen - dCurrLen) / dLen
|
||||
end
|
||||
end
|
||||
|
||||
EgtModifyCurveStartPoint( nNewEntId, EgtSP( nNewEntId, GDB_ID.ROOT) + vDelta[0] * vtSlicing, GDB_RT.GLOB)
|
||||
for dU = 1, dE - 1 do
|
||||
local ptCurr = EgtUP( nNewEntId, dU, GDB_ID.ROOT)
|
||||
EgtModifyCurveCompoJoint( nNewEntId, dU, ptCurr + vDelta[dU] * vtSlicing, GDB_RT.GLOB)
|
||||
end
|
||||
|
||||
-- alzo l'ultimo tratto del layer precedente con una proiezione ortogonale "graduale" sul piano corrente
|
||||
-- ricavo il tratto sulla curva precedente da alzare
|
||||
local dTrimPar = EgtCurveParamAtLength( nOldId, EgtCurveLength( nOldId) - s_dSpralVaseFirstDelta)
|
||||
local nLastId = EgtSplitCurveAtParam( nOldId, dTrimPar)
|
||||
EgtApproxCurve( nLastId, GDB_CA.SPECIAL_LINES, 0.01, s_dSpiralVaseMaxLen)
|
||||
-- alzo in maniera graduale
|
||||
local _, dEOld = EgtCurveDomain( nLastId)
|
||||
local vDeltaOld = {}
|
||||
local dLenOld = EgtCurveLength( nLastId)
|
||||
for dU = 0, dEOld do
|
||||
local dCurrLen = EgtCurveLengthAtParam( nLastId, dU)
|
||||
vDeltaOld[dU] = abs( vDelta[0]) * dCurrLen / dLenOld
|
||||
end
|
||||
for dU = 0, dEOld - 1 do
|
||||
local ptCurr = EgtUP( nLastId, dU, GDB_ID.ROOT)
|
||||
EgtModifyCurveCompoJoint( nLastId, dU, ptCurr + vDeltaOld[dU] * vtSlicing, GDB_RT.GLOB)
|
||||
end
|
||||
EgtModifyCurveEndPoint( nLastId, EgtEP( nLastId, GDB_ID.ROOT) + vDeltaOld[dEOld] * vtSlicing, GDB_RT.GLOB)
|
||||
|
||||
-- approssimo
|
||||
EgtApproxCurve( nLastId, GDB_CA.ARCS, s_dApproxTol)
|
||||
if LayerParams.bLinearApprox then
|
||||
EgtApproxCurve( nLastId, GDB_CA.LINES, LayerParams.dLinearApproxTol)
|
||||
end
|
||||
EgtModifyCurveEndPoint( nLastId, EgtSP( nNewEntId, GDB_ID.ROOT), GDB_RT.GLOB)
|
||||
|
||||
else
|
||||
|
||||
local dLen = EgtCurveLength( nNewEntId)
|
||||
local _, dE = EgtCurveDomain( nNewEntId)
|
||||
local vDelta = {}
|
||||
for dU = 0, dE - 1 do
|
||||
local dCurrLen = EgtCurveLengthAtParam( nNewEntId, dU)
|
||||
local ptCurr = EgtUP( nNewEntId, dU, GDB_ID.ROOT)
|
||||
local dDist = ( ( ptCurr - ptSlicingPrev) * vtSlicingPrev) / dCosAng -- distanza dal piano precedente lungo vtSlicing
|
||||
vDelta[dU] = - dDist * ( dLen - dCurrLen) / dLen
|
||||
end
|
||||
|
||||
EgtModifyCurveStartPoint( nNewEntId, EgtSP( nNewEntId, GDB_ID.ROOT) + vDelta[0] * vtSlicing, GDB_RT.GLOB)
|
||||
for dU = 1, dE - 1 do
|
||||
local ptCurr = EgtUP( nNewEntId, dU, GDB_ID.ROOT)
|
||||
EgtModifyCurveCompoJoint( nNewEntId, dU, ptCurr + vDelta[dU] * vtSlicing, GDB_RT.GLOB)
|
||||
end
|
||||
end
|
||||
|
||||
-- c) approssimo
|
||||
local ptS = EgtSP( nNewEntId, GDB_ID.ROOT)
|
||||
EgtApproxCurve( nNewEntId, GDB_CA.ARCS, s_dApproxTol)
|
||||
if LayerParams.bLinearApprox then
|
||||
EgtApproxCurve( nNewEntId, GDB_CA.LINES, LayerParams.dLinearApproxTol)
|
||||
end
|
||||
if EgtCurveIsClosed( nNewEntId) then
|
||||
EgtChangeClosedCurveStartPoint( nNewEntId, ptS, GDB_RT.GLOB)
|
||||
end
|
||||
end
|
||||
|
||||
local vTPathsCrvs = EgtGetAllInGroup( nTpathGrpId)
|
||||
nOldId = vTPathsCrvs[#vTPathsCrvs]
|
||||
nOldPathId = vEntIds[1]
|
||||
nRealLayer = nRealLayer + 1
|
||||
end
|
||||
|
||||
if EgtProcessEvents( EgtIf( PRINT, 300, 0) + nIdx / #vLayIds * 100, 0) == 1 then
|
||||
EgtDraw()
|
||||
return false
|
||||
end
|
||||
end
|
||||
|
||||
-- creo un un layer extra piano alla quota finale
|
||||
local nLastLay = EgtCopyGlob( vLayIds[#vLayIds], vLayIds[#vLayIds], GDB_IN.AFTER)
|
||||
-- aggiorno nome e numero slice
|
||||
local nSliceNbr = EgtGetInfo( nLastLay, KEY_SLICE_NBR, 'i') + 1
|
||||
EgtSetName( nLastLay, SLICE_LAYER .. EgtNumToString( nSliceNbr))
|
||||
EgtSetInfo( nLastLay, KEY_SLICE_NBR, nSliceNbr)
|
||||
-- creo la curva di toolpath alla quota finale
|
||||
local nCrvGrpId = EgtGetFirstNameInGroup( nLastLay, CONTOUR_GRP.."*")
|
||||
local nPathGrpId = EgtGetFirstNameInGroup( nCrvGrpId, PATH_GRP)
|
||||
local nTpathGrpId = EgtGetFirstNameInGroup( nCrvGrpId, TOOLPATH_GRP)
|
||||
EgtEmptyGroup( nTpathGrpId)
|
||||
local nId = EgtCopyGlob( EgtGetFirstInGroup( nPathGrpId), nTpathGrpId)
|
||||
local vtSlicing = EgtGetInfo( nLastLay, KEY_SLICE_DIR, 'v')
|
||||
EgtMove( nId, LayerParams.dLayHeight * vtSlicing, GDB_RT.GLOB)
|
||||
|
||||
local ptS = EgtEP( nOldId, GDB_ID.ROOT)
|
||||
EgtChangeClosedCurveStartPoint( nId, ptS, GDB_RT.GLOB)
|
||||
EgtModifyCurveExtrusion( nId, vtSlicing, GDB_RT.GLOB)
|
||||
EgtSetInfo( nId, KEY_CRV_STRAND, LayerParams.dStrand)
|
||||
if LayerParams.bInvert then
|
||||
EgtInvertCurve( nId)
|
||||
EgtSetInfo( nId, KEY_INVERTED_CRV, 1)
|
||||
end
|
||||
EgtSetColor( nId, EgtStdColor('GRAY'))
|
||||
|
||||
-- approssimo
|
||||
EgtApproxCurve( nId, GDB_CA.ARCS, s_dApproxTol)
|
||||
if LayerParams.bLinearApprox then
|
||||
EgtApproxCurve( nId, GDB_CA.LINES, LayerParams.dLinearApproxTol)
|
||||
end
|
||||
if EgtCurveIsClosed( nId) then
|
||||
EgtChangeClosedCurveStartPoint( nId, ptS, GDB_RT.GLOB)
|
||||
end
|
||||
|
||||
return true
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function SpiralVaseFull( vLayIds, LayerParams)
|
||||
-- la differenza in altezza tra due layers viene distribuita uniformemente lungo tutto il percorso
|
||||
-- la continuità tra i layers viene risolta sul layer corrente ( uniformemente su tutto il tratto o solo nella parte iniziale di lunghezza dSpiralVaseInterpLen)
|
||||
-- gestione speciale dei primi layers per gestione dell'altezza e feed
|
||||
|
||||
local nSlicingType = EgtGetInfo( s_nPartId, KEY_SLICING_TYPE, 'i')
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
return SpiralVaseFullMultiPlanar( vLayIds, LayerParams)
|
||||
end
|
||||
|
||||
local nOldId, nOldPathId
|
||||
local nRealLayer = 1
|
||||
local nSlicingType = EgtGetInfo( s_nPartId, KEY_SLICING_TYPE, 'i')
|
||||
|
||||
-- copio ultimo layer
|
||||
if nSlicingType ~= SLICING_TYPE.DEG45_X and nSlicingType ~= SLICING_TYPE.DEG45_Y then
|
||||
local nNewLay = EgtCopyGlob( vLayIds[#vLayIds], vLayIds[#vLayIds], GDB_IN.AFTER)
|
||||
table.insert( vLayIds, nNewLay)
|
||||
-- aggiorno nome e numero slice
|
||||
local nSliceNbr = EgtGetInfo( nNewLay, KEY_SLICE_NBR, 'i') + 1
|
||||
EgtSetName( nNewLay, SLICE_LAYER .. EgtNumToString( nSliceNbr))
|
||||
EgtSetInfo( nNewLay, KEY_SLICE_NBR, nSliceNbr)
|
||||
-- porto alla nuova quota
|
||||
EgtMove( nNewLay, LayerParams.dLayHeight * LayerParams.vtSlicing, GDB_RT.GLOB)
|
||||
-- aggiorno nome e info
|
||||
local sOldIdx = string.sub( EgtGetName( vLayIds[#vLayIds]), 6)
|
||||
local nNewIdx = tonumber( sOldIdx) + 1
|
||||
EgtSetName( nNewLay, SLICE_LAYER .. EgtNumToString( nNewIdx))
|
||||
EgtSetInfo( nNewLay, KEY_SLICE_NBR, nNewIdx)
|
||||
local dZOld = EgtGetInfo( vLayIds[#vLayIds], KEY_SLICE_Z, 'd') or 0
|
||||
EgtSetInfo( nNewLay, KEY_SLICE_Z, dZOld + LayerParams.dLayHeight)
|
||||
local ptSlice = EgtGetInfo( nNewLay, KEY_SLICE_POS, 'p')
|
||||
ptSlice = ptSlice + LayerParams.dLayHeight * LayerParams.vtSlicing
|
||||
EgtSetInfo( nNewLay, KEY_SLICE_POS, ptSlice)
|
||||
end
|
||||
|
||||
-- ciclo sui layer
|
||||
@@ -2228,6 +2545,194 @@ local function SpiralVaseFull( vLayIds, LayerParams)
|
||||
|
||||
nRealLayer = nRealLayer + 1
|
||||
end
|
||||
|
||||
if EgtProcessEvents( EgtIf( PRINT, 300, 0) + nIdx / #vLayIds * 100, 0) == 1 then
|
||||
EgtDraw()
|
||||
return false
|
||||
end
|
||||
end
|
||||
|
||||
return true
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function SpiralVasePartialMultiPlanar( vLayIds, LayerParams)
|
||||
-- la differenza in altezza tra due layers viene distribuita solo lungo il tratto finale di lunghezza dSpiralVaseLen
|
||||
-- la continuità tra i layers viene risolta sul tratto finale del layer percedente
|
||||
|
||||
local bFirst = true
|
||||
-- individuo l'ultimo layer ( alcuni potrebbero essere vuoti)
|
||||
local nLastLay = 1
|
||||
for nIdx = #vLayIds, 1, -1 do
|
||||
local nCrvGrp = EgtGetFirstNameInGroup( vLayIds[ nIdx], CONTOUR_GRP.."*")
|
||||
local nPathGrpId = EgtGetFirstNameInGroup( nCrvGrp, PATH_GRP) or GDB_ID_NULL
|
||||
if EgtGetGroupObjs( nPathGrpId) > 0 then
|
||||
nLastLay = nIdx
|
||||
break
|
||||
end
|
||||
end
|
||||
|
||||
-- ciclo sui layer
|
||||
local nOldId
|
||||
for nIdx = 1, #vLayIds do
|
||||
|
||||
-- piano di slicing
|
||||
local vtSlicing = EgtGetInfo( vLayIds[nIdx], KEY_SLICE_DIR, 'v')
|
||||
local ptSlicing = EgtGetInfo( vLayIds[nIdx], KEY_SLICE_POS, 'p')
|
||||
|
||||
-- cerco i gruppi di contorni
|
||||
local vCrvGrpIds = EgtGetNameInGroup( vLayIds[ nIdx], CONTOUR_GRP.."*")
|
||||
if #vCrvGrpIds > 1 then
|
||||
-- se più di un gruppo di curve errore
|
||||
EgtOutBox( 'Error in spiral vase : layer ' .. tostring( nIdx) .. ' has more than one toolpath', 'ToolPathCalc')
|
||||
return false
|
||||
end
|
||||
|
||||
-- recupero il gruppo dei percorsi
|
||||
local nPathGrpId = EgtGetFirstNameInGroup( vCrvGrpIds[1], PATH_GRP)
|
||||
if not nPathGrpId then
|
||||
EgtOutBox( 'Error missing paths', 'ToolPathCalc')
|
||||
return false
|
||||
else
|
||||
EgtSetStatus( nPathGrpId, GDB_ST.OFF)
|
||||
end
|
||||
|
||||
-- recupero il gruppo dei percorsi utensile
|
||||
local nTpathGrpId = EgtGetFirstNameInGroup( vCrvGrpIds[1], TOOLPATH_GRP)
|
||||
if not nTpathGrpId then
|
||||
nTpathGrpId = EgtGroup( vCrvGrpIds[1])
|
||||
EgtSetName( nTpathGrpId, TOOLPATH_GRP)
|
||||
else
|
||||
EgtEmptyGroup( nTpathGrpId)
|
||||
end
|
||||
|
||||
-- creo il percorso di lavoro
|
||||
local vEntIds = EgtGetAllInGroup( nPathGrpId)
|
||||
if #vEntIds > 1 then
|
||||
EgtOutBox( 'Error in spiral vase : layer ' .. tostring( nIdx) .. ' has more than one toolpath', 'ToolPathCalc')
|
||||
return false
|
||||
end
|
||||
|
||||
local nNewEntId = EgtCopyGlob( vEntIds[1] or GDB_ID.NULL, nTpathGrpId, GDB_IN.LAST_SON)
|
||||
if nNewEntId then
|
||||
|
||||
EgtModifyCurveExtrusion( nNewEntId, vtSlicing, GDB_RT.GLOB)
|
||||
EgtSetInfo( nNewEntId, KEY_CRV_STRAND, LayerParams.dStrand)
|
||||
if LayerParams.bInvert then
|
||||
EgtInvertCurve( nNewEntId)
|
||||
EgtSetInfo( nNewEntId, KEY_INVERTED_CRV, 1)
|
||||
end
|
||||
EgtSetColor( nNewEntId, EgtStdColor('GRAY'))
|
||||
|
||||
-- sistemo il tratto finale del percorso precedente
|
||||
if not bFirst then
|
||||
|
||||
local ptSOld = EgtSP( nOldId, GDB_ID.ROOT)
|
||||
local ptEOld = EgtEP( nOldId, GDB_ID.ROOT)
|
||||
-- proiezione ortogonale dei punti sul piano di slicing corrente
|
||||
local ptSProj = ptSOld - ( ( ptSOld - ptSlicing) * vtSlicing) * vtSlicing
|
||||
local ptEProj = ptEOld - ( ( ptEOld - ptSlicing) * vtSlicing) * vtSlicing
|
||||
|
||||
-- a) continuità tra i layers
|
||||
-- modifico il punto iniziale corrente per avvicinarmi il più possibile alla fine del percorso precedente
|
||||
EgtChangeClosedCurveStartPoint( nNewEntId, ptEProj, GDB_RT.GLOB)
|
||||
local ptCurrStart = EgtSP( nNewEntId, GDB_ID.ROOT)
|
||||
if not AreSamePointApprox( ptEProj, ptCurrStart) then
|
||||
-- se i percorsi non sono in continuità modifico il tratto finale del percorso precedente in modo da farlo arrivare sul punto di inizio del percorso corrente
|
||||
-- usando come guida la proiezione obliqua del percorso corrente sul piano precedente ( SpiralizeAlongGuide lavora con curve piane complanari)
|
||||
|
||||
local nProjCrv = EgtCopyGlob( vEntIds[1], nPathGrpId)
|
||||
if LayerParams.bInvert then
|
||||
EgtInvertCurve( nProjCrv)
|
||||
end
|
||||
-- conservo solo il sottotratto corrispondente a nOldId
|
||||
local _, _, dStartGuide = EgtPointCurveDist( ptSProj, nProjCrv, GDB_ID.ROOT)
|
||||
local _, _, dEndGuide = EgtPointCurveDist( ptEProj, nProjCrv, GDB_ID.ROOT)
|
||||
EgtTrimCurveStartEndAtParam( nProjCrv, dStartGuide, dEndGuide)
|
||||
-- proiezione sul piano precedente ( ptSlicingPrev, vtSlicingPrev) lungo vtSlicing
|
||||
local vtSlicingPrev = EgtGetInfo( vLayIds[nIdx-1], KEY_SLICE_DIR, 'v')
|
||||
local ptSlicingPrev = EgtGetInfo( vLayIds[nIdx-1], KEY_SLICE_POS, 'p') + LayerParams.dLayHeight * vtSlicingPrev
|
||||
local _, dE = EgtCurveDomain( nProjCrv)
|
||||
for dU = 0, dE do
|
||||
local pt = EgtUP( nProjCrv, dU, GDB_ID.ROOT)
|
||||
local dDist = ( pt - ptSlicingPrev) * vtSlicingPrev / ( vtSlicingPrev * vtSlicing)
|
||||
EgtModifyCurveCompoJoint( nProjCrv, dU, pt - dDist * vtSlicing, GDB_RT.GLOB)
|
||||
end
|
||||
-- approssimazione dei tratti per migliorare la qualità dello spiralize
|
||||
EgtApproxCurve( nProjCrv, GDB_CA.SPECIAL_LINES, 0.01, s_dSpiralVaseMaxLen)
|
||||
EgtApproxCurve( nOldId, GDB_CA.SPECIAL_LINES, 0.01, s_dSpiralVaseMaxLen)
|
||||
-- spiralize
|
||||
EgtInvertCurve( nProjCrv)
|
||||
EgtInvertCurve( nOldId)
|
||||
EgtSpiralizeCurveAlongGuide( nOldId, nProjCrv)
|
||||
EgtInvertCurve( nOldId)
|
||||
|
||||
EgtErase( nProjCrv)
|
||||
end
|
||||
|
||||
-- b) proiezione ortogonale graduale sul piano corrente
|
||||
-- delta che va applicato ( è la distanza ortogonale del punto finale di nOldId dal piano del toolpath corrente)
|
||||
local dDist = ( EgtEP( nOldId, GDB_ID.ROOT) - ( ptSlicing + LayerParams.dLayHeight * vtSlicing)) * vtSlicing
|
||||
EgtApproxCurve( nOldId, GDB_CA.SPECIAL_LINES, 0.01, s_dSpiralVaseMaxLen)
|
||||
local _, dE = EgtCurveDomain( nOldId)
|
||||
local vDelta = {}
|
||||
local dLen = EgtCurveLength( nOldId)
|
||||
for dU = 0, dE do
|
||||
local dCurrLen = EgtCurveLengthAtParam( nOldId, dU)
|
||||
vDelta[dU] = abs( dDist) * dCurrLen / dLen
|
||||
end
|
||||
for dU = 0, dE do
|
||||
local ptCurr = EgtUP( nOldId, dU, GDB_ID.ROOT)
|
||||
EgtModifyCurveCompoJoint( nOldId, dU, ptCurr + vDelta[dU] * vtSlicing, GDB_RT.GLOB)
|
||||
end
|
||||
|
||||
-- c) approssimazione della curva
|
||||
EgtApproxCurve( nOldId, GDB_CA.ARCS, s_dApproxTol)
|
||||
if LayerParams.bLinearApprox then
|
||||
EgtApproxCurve( nOldId, GDB_CA.LINES, LayerParams.dLinearApproxTol)
|
||||
end
|
||||
-- forzo punti per garantire continuità
|
||||
EgtModifyCurveStartPoint( nOldId, ptSOld, GDB_RT.GLOB)
|
||||
EgtModifyCurveEndPoint( nOldId, ptCurrStart + LayerParams.dLayHeight * vtSlicing, GDB_RT.GLOB)
|
||||
end
|
||||
|
||||
-- spostamento dell'altezza layer
|
||||
local vtMove = LayerParams.dLayHeight * vtSlicing
|
||||
EgtMove( nNewEntId, vtMove, GDB_RT.GLOB)
|
||||
EgtCopyGlob( nNewEntId, nPathGrpId)
|
||||
|
||||
-- approssimazione
|
||||
if LayerParams.bLinearApprox then
|
||||
local ptS = EgtSP( nNewEntId, GDB_ID.ROOT)
|
||||
EgtApproxCurve( nNewEntId, GDB_CA.LINES, LayerParams.dLinearApproxTol)
|
||||
if EgtCurveIsClosed( nNewEntId) then
|
||||
EgtChangeClosedCurveStartPoint( nNewEntId, ptS, GDB_RT.GLOB)
|
||||
end
|
||||
end
|
||||
|
||||
-- eventuale lead in
|
||||
if bFirst then
|
||||
bFirst = false
|
||||
if LayerParams.nLeadInType ~= LEAD_TYPE.NONE then
|
||||
EgtTrimCurveStartAtLen( nNewEntId, LayerParams.dOffsetLP)
|
||||
local nLeadInCrv = AddLeadIn( nNewEntId, LayerParams, nTpathGrpId)
|
||||
if nLeadInCrv and LayerParams.bLinearApprox then
|
||||
EgtApproxCurve( nLeadInCrv, GDB_CA.LINES, LayerParams.dLinearApproxTol)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
-- separo l'ultimo tratto che farà da collegamento con il layer successivo
|
||||
if nIdx ~= nLastLay then
|
||||
local dTrimLen = EgtCurveLength( nNewEntId) - LayerParams.dSpiralVaseLen
|
||||
if dTrimLen < GEO.EPS_SMALL then
|
||||
-- se lunghezza di trim maggiore delle dimensioni forzo un valore accettabile
|
||||
dTrimLen = 0.5 * EgtCurveLength( nNewEntId)
|
||||
end
|
||||
local dTrimPar = EgtCurveParamAtLength( nNewEntId, dTrimLen)
|
||||
nOldId = EgtSplitCurveAtParam( nNewEntId, dTrimPar)
|
||||
end
|
||||
end
|
||||
|
||||
if EgtProcessEvents( EgtIf( PRINT, 300, 0) + nIdx / #vLayIds * 100, 0) == 1 then
|
||||
EgtDraw()
|
||||
@@ -2243,6 +2748,11 @@ local function SpiralVasePartial( vLayIds, LayerParams)
|
||||
-- la differenza in altezza tra due layers viene distribuita solo lungo il tratto finale di lunghezza dSpiralVaseLen
|
||||
-- la continuità tra i layers viene risolta sul tratto finale del layer percedente
|
||||
|
||||
local nSlicingType = EgtGetInfo( s_nPartId, KEY_SLICING_TYPE, 'i')
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
return SpiralVasePartialMultiPlanar( vLayIds, LayerParams)
|
||||
end
|
||||
|
||||
local bFirst = true
|
||||
-- individuo l'ultimo layer ( alcuni potrebbero essere vuoti)
|
||||
local nLastLay = 1
|
||||
@@ -2409,23 +2919,16 @@ local function SpiralVase( vLayIds, LayerParams)
|
||||
return false
|
||||
end
|
||||
end
|
||||
|
||||
-- calcolo il box dei percorsi
|
||||
local b3Tot = BBox3d()
|
||||
local nLastTPath
|
||||
for i = 1, #vLayIds do
|
||||
local vCrvGrpIds = EgtGetNameInGroup( vLayIds[i], CONTOUR_GRP.."*")
|
||||
local nTpathGrpId = EgtGetFirstNameInGroup( vCrvGrpIds[1], TOOLPATH_GRP)
|
||||
nLastTPath = EgtGetLastInGroup( nTpathGrpId)
|
||||
local b3Box = ComputeToolPathBox( nTpathGrpId)
|
||||
b3Tot:Add( b3Box)
|
||||
end
|
||||
|
||||
|
||||
-- aggiungo uscita, coasting e wipe su ultima curva
|
||||
local nLastLayer = EgtGetLastNameInGroup( s_nPartId, SLICE_LAYER .. '*')
|
||||
local vCrvGrpIds = EgtGetNameInGroup( nLastLayer, CONTOUR_GRP.."*")
|
||||
local nTpathGrpId = EgtGetFirstNameInGroup( vCrvGrpIds[1], TOOLPATH_GRP)
|
||||
local nLastTPath = EgtGetLastInGroup( nTpathGrpId)
|
||||
AddSpiralVaseLeadOut( nLastTPath, LayerParams)
|
||||
|
||||
-- correzione in z
|
||||
AddZCorrection( b3Tot, LayerParams)
|
||||
AddZCorrection( LayerParams)
|
||||
|
||||
return true
|
||||
end
|
||||
@@ -2444,19 +2947,23 @@ function CalcToolPath.Exec( nPartId)
|
||||
|
||||
-- recupero i parametri per calcolo dei toolpath
|
||||
local LayerParams = GetLayerParamsForToolPathCalc()
|
||||
local nSlicingType = EgtGetInfo( s_nPartId, KEY_SLICING_TYPE, 'i')
|
||||
|
||||
-- caso spiral vase
|
||||
if LayerParams.bSpiralVase then
|
||||
return SpiralVase( vLayIds, LayerParams)
|
||||
end
|
||||
|
||||
local b3Tot = BBox3d() -- box dei toolpath
|
||||
local nLayCnt = 1
|
||||
-- Ciclo sui layer
|
||||
for nIdx = 1, #vLayIds do
|
||||
|
||||
s_nCurrIdx = nIdx
|
||||
|
||||
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
LayerParams.vtSlicing = EgtGetInfo( vLayIds[nIdx], KEY_SLICE_DIR, 'v')
|
||||
end
|
||||
|
||||
-- scorro tutti i gruppi di contorni
|
||||
local nCrvGrpId = EgtGetFirstNameInGroup( vLayIds[ nIdx], CONTOUR_GRP.."*")
|
||||
while nCrvGrpId do
|
||||
@@ -2532,10 +3039,6 @@ function CalcToolPath.Exec( nPartId)
|
||||
end
|
||||
end
|
||||
|
||||
-- aggiorno il box dei toolpath
|
||||
local b3Box = ComputeToolPathBox( nTpathGrpId)
|
||||
b3Tot:Add( b3Box)
|
||||
|
||||
-- verifico non sia vuoto per aggiornare il numero di layers
|
||||
if EgtGetFirstInGroup( nTpathGrpId) then
|
||||
nLayCnt = nIdx
|
||||
@@ -2561,7 +3064,7 @@ function CalcToolPath.Exec( nPartId)
|
||||
end
|
||||
|
||||
-- correzione in z
|
||||
AddZCorrection( b3Tot, LayerParams)
|
||||
AddZCorrection( LayerParams)
|
||||
|
||||
-- setto info con numero di layers effettivi
|
||||
EgtSetInfo( nPartId, KEY_LAYER_CNT, nLayCnt)
|
||||
|
||||
+536
-82
@@ -15,13 +15,16 @@ local AMD = require( 'AddManData')
|
||||
--------------------------------------------------------------------
|
||||
local s_dTol = 0.1
|
||||
local s_nSimplifiedSection = 0
|
||||
local s_dMultiPlanarH = 0
|
||||
local s_dColorFactor = 0.9
|
||||
local s_nPartId
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function GetLayerParamsForSolidCalc( nPartId)
|
||||
local LayerParams = {}
|
||||
LayerParams.bSpiralVase = EgtGetInfo( nPartId, KEY_SPIRAL_VASE, 'b') or false
|
||||
LayerParams.dLayHeight = EgtGetInfo( nPartId, KEY_SLICE_STEP, 'd')
|
||||
LayerParams.vtSlicing = EgtGetInfo( nPartId, KEY_SLICING_DIR, 'v')
|
||||
LayerParams.vtSlicing = EgtGetInfo( nPartId, KEY_SLICE_DIR, 'v') or EgtGetInfo( nPartId, 'SlicingDir', 'v')
|
||||
LayerParams.dStrand = EgtGetInfo( nPartId, KEY_STRAND, 'd')
|
||||
return LayerParams
|
||||
end
|
||||
@@ -39,10 +42,10 @@ local function CalcSectionParams( dStrand, dH)
|
||||
end
|
||||
|
||||
----------------------------------------------------------------------
|
||||
local function CreateStandardSolid( nCrvId, nSolidGrp, LayerParams, dStrand)
|
||||
|
||||
local dBevelX, dBevelY = CalcSectionParams( dStrand, LayerParams.dLayHeight)
|
||||
local nSrfId = EgtSurfTmRectSwept( nSolidGrp, dStrand, LayerParams.dLayHeight, dBevelX, dBevelY, nCrvId, GDB_RSCT.BEVEL, s_dTol)
|
||||
local function CreateStandardSolid( nCrvId, nSolidGrp, dH, dStrand)
|
||||
|
||||
local dBevelX, dBevelY = CalcSectionParams( dStrand, dH)
|
||||
local nSrfId = EgtSurfTmRectSwept( nSolidGrp, dStrand, dH, dBevelX, dBevelY, nCrvId, GDB_RSCT.BEVEL, s_dTol)
|
||||
return nSrfId
|
||||
end
|
||||
|
||||
@@ -105,9 +108,7 @@ local function CreateSpiralVaseCap( nSectId, vtDir, nSolidGrp)
|
||||
local nCapSrf = EgtSurfTmByTriangles( nSolidGrp, vSurfs)
|
||||
|
||||
-- cancello curve di costruzione
|
||||
for i = 1, #vCrvs do
|
||||
EgtErase( vCrvs[i])
|
||||
end
|
||||
EgtErase( vCrvs)
|
||||
EgtErase( nCrvTop)
|
||||
EgtErase( nCrvBottom)
|
||||
|
||||
@@ -115,10 +116,10 @@ local function CreateSpiralVaseCap( nSectId, vtDir, nSolidGrp)
|
||||
end
|
||||
|
||||
-------------------------------------------------------------------------------
|
||||
local function CreateSpiralVaseSolid( nCrvId, nSolidGrp, LayerParams, dStrand)
|
||||
local function CreateSpiralVaseSolid( nCrvId, nSolidGrp, vtSlicing, dH, dStrand)
|
||||
|
||||
-- gruppo temporaneo per conti
|
||||
local nGrpTmp = EgtGroup( nSolidGrp, Frame3d( ORIG(), LayerParams.vtSlicing, GDB_RT.GLOB))
|
||||
local nGrpTmp = EgtGroup( nSolidGrp, Frame3d( ORIG(), vtSlicing, GDB_RT.GLOB))
|
||||
|
||||
-- accorcio leggermente la curva per evitare problemi di inconsistent orientation nel solido
|
||||
local dLen = EgtCurveLength( nCrvId)
|
||||
@@ -128,28 +129,28 @@ local function CreateSpiralVaseSolid( nCrvId, nSolidGrp, LayerParams, dStrand)
|
||||
local vtS = EgtSV( nCrvId, GDB_ID.ROOT)
|
||||
local ptE = EgtEP( nCrvId, GDB_ID.ROOT)
|
||||
local vtE = EgtEV( nCrvId, GDB_ID.ROOT)
|
||||
local dDelta = ( ptE - ptS) * LayerParams.vtSlicing
|
||||
local dDelta = ( ptE - ptS) * vtSlicing
|
||||
-- se non è vero spiral vase, chiamo funzione standard
|
||||
if dDelta < GEO.EPS_SMALL then
|
||||
EgtErase( nGrpTmp)
|
||||
return CreateStandardSolid( nCrvId, nSolidGrp, LayerParams, dStrand)
|
||||
return CreateStandardSolid( nCrvId, nSolidGrp, dH, dStrand)
|
||||
end
|
||||
|
||||
-- appiattisco la curva
|
||||
local nCrvCopy = EgtCopyGlob( nCrvId, nGrpTmp)
|
||||
EgtModifyCurveExtrusion( nCrvCopy, LayerParams.vtSlicing, GDB_RT.GLOB)
|
||||
EgtProjectCurveOnPlane( nCrvCopy, ptS, LayerParams.vtSlicing, GDB_RT.GLOB)
|
||||
EgtModifyCurveExtrusion( nCrvCopy, vtSlicing, GDB_RT.GLOB)
|
||||
EgtProjectCurveOnPlane( nCrvCopy, ptS, vtSlicing, GDB_RT.GLOB)
|
||||
EgtMergeCurvesInCurveCompo( nCrvCopy)
|
||||
EgtChangeClosedCurveStartPoint( nCrvCopy, ptS, GDB_RT.GLOB)
|
||||
|
||||
-- calcolo la sezione iniziale
|
||||
local vtDir = EgtSV( nCrvCopy, GDB_ID.ROOT)
|
||||
vtDir:rotate( LayerParams.vtSlicing, 90)
|
||||
local nSectId = CreateSection( ptS, vtDir, dStrand, LayerParams.dLayHeight, LayerParams.vtSlicing, nGrpTmp)
|
||||
vtDir:rotate( vtSlicing, 90)
|
||||
local nSectId = CreateSection( ptS, vtDir, dStrand, dH, vtSlicing, nGrpTmp)
|
||||
-- creo la sezione finale
|
||||
local vtDir2 = EgtEV( nCrvCopy, GDB_ID.ROOT)
|
||||
vtDir2:rotate( LayerParams.vtSlicing, 90)
|
||||
local nSectE = CreateSection( ptE, vtDir2, dStrand, LayerParams.dLayHeight, LayerParams.vtSlicing, nGrpTmp)
|
||||
vtDir2:rotate( vtSlicing, 90)
|
||||
local nSectE = CreateSection( ptE, vtDir2, dStrand, dH, vtSlicing, nGrpTmp)
|
||||
|
||||
-- creo il solido aperto (tubo)
|
||||
local vCrvs = {}
|
||||
@@ -162,8 +163,8 @@ local function CreateSpiralVaseSolid( nCrvId, nSolidGrp, LayerParams, dStrand)
|
||||
EgtErase( nGrpTmp)
|
||||
return nil
|
||||
end
|
||||
local dMove = ( ptRef - ptS) * LayerParams.vtSlicing
|
||||
EgtMove( vCrvs[i+1], LayerParams.vtSlicing * dMove, GDB_RT.GLOB)
|
||||
local dMove = ( ptRef - ptS) * vtSlicing
|
||||
EgtMove( vCrvs[i+1], vtSlicing * dMove, GDB_RT.GLOB)
|
||||
EgtSpiralizeCurveAlongExtrusion( vCrvs[i+1], dDelta)
|
||||
|
||||
-- modifico la curva per congiungerla ai caps
|
||||
@@ -186,32 +187,32 @@ local function CreateSpiralVaseSolid( nCrvId, nSolidGrp, LayerParams, dStrand)
|
||||
return nil
|
||||
end
|
||||
end
|
||||
|
||||
local nSrfId = EgtSurfTmByTriangles( nSolidGrp, vSurfs)
|
||||
|
||||
|
||||
-- creazione del mezzo disco iniziale
|
||||
local nCap1 = CreateSpiralVaseCap( nSectId, - vtS, nSolidGrp)
|
||||
vSurfs[#vCrvs] = CreateSpiralVaseCap( nSectId, - vtS, nSolidGrp)
|
||||
-- creazione del mezzo disco finale
|
||||
local nCap2 = CreateSpiralVaseCap( nSectE, vtE, nSolidGrp)
|
||||
EgtInvertSurf( nCap2)
|
||||
vSurfs[#vCrvs+1] = CreateSpiralVaseCap( nSectE, vtE, nSolidGrp)
|
||||
EgtInvertSurf( vSurfs[#vCrvs+1])
|
||||
|
||||
local nSolidId = EgtSurfTmByTriangles( nSolidGrp, vSurfs)
|
||||
|
||||
-- cancello le curve usate per la costruzione
|
||||
EgtErase( nGrpTmp)
|
||||
|
||||
return EgtSurfTmByTriangles( nSolidGrp, { nSrfId, nCap1, nCap2})
|
||||
return nSolidId
|
||||
end
|
||||
|
||||
--------------------------------------------------------------------------------------
|
||||
local function CreateSolid( nCrvId, nSolidGrp, LayerParams, dStrand)
|
||||
local function CreateSolid( nCrvId, nSolidGrp, LayerParams, dH, dStrand)
|
||||
if LayerParams.bSpiralVase then
|
||||
return CreateSpiralVaseSolid( nCrvId, nSolidGrp, LayerParams, dStrand)
|
||||
return CreateSpiralVaseSolid( nCrvId, nSolidGrp, LayerParams.vtSlicing, dH, dStrand)
|
||||
else
|
||||
return CreateStandardSolid( nCrvId, nSolidGrp, LayerParams, dStrand)
|
||||
return CreateStandardSolid( nCrvId, nSolidGrp, dH, dStrand)
|
||||
end
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function CreateDirectionArrow( nCrvId, nSolidGrp, vtSlicing, dStrand, nLayer)
|
||||
local function CreateDirectionArrow( nCrvId, nSolidGrp, vtSlicing, dStrand, nSliceNbr)
|
||||
|
||||
local ptS = EgtSP( nCrvId, GDB_RT.GLOB)
|
||||
local vtS = EgtSV( nCrvId, GDB_RT.GLOB)
|
||||
@@ -233,7 +234,7 @@ local function CreateDirectionArrow( nCrvId, nSolidGrp, vtSlicing, dStrand, nLay
|
||||
local nSrf = EgtSurfFlatRegion( nSolidGrp, { nCompo})
|
||||
EgtErase( nCompo)
|
||||
EgtSetColor( nSrf, RED())
|
||||
EgtSetInfo( nSrf, KEY_SLICE_NBR, nLayer)
|
||||
EgtSetInfo( nSrf, KEY_SLICE_NBR, nSliceNbr)
|
||||
EgtSetStatus( nSrf, GDB_ST.OFF)
|
||||
EgtSetMode( nSrf, GDB_MD.HIDDEN)
|
||||
EgtSetName( nSrf, DIR_ARROW)
|
||||
@@ -241,10 +242,10 @@ local function CreateDirectionArrow( nCrvId, nSolidGrp, vtSlicing, dStrand, nLay
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function CreateRecursiveSolid( nCrvId, vSurfs, nSolidGrp, LayerParams, dStrand)
|
||||
local function CreateRecursiveSolid( nCrvId, vSurfs, nSolidGrp, LayerParams, dH, dStrand)
|
||||
|
||||
-- tento la creazione del solido
|
||||
local nSurf = CreateSolid( nCrvId, nSolidGrp, LayerParams, dStrand - 50 * GEO.EPS_SMALL)
|
||||
local nSurf = CreateSolid( nCrvId, nSolidGrp, LayerParams, dH, dStrand - 50 * GEO.EPS_SMALL)
|
||||
if nSurf then
|
||||
EgtErase( nCrvId)
|
||||
table.insert( vSurfs, nSurf)
|
||||
@@ -264,16 +265,66 @@ local function CreateRecursiveSolid( nCrvId, vSurfs, nSolidGrp, LayerParams, dSt
|
||||
EgtErase( nCrvId)
|
||||
return false
|
||||
end
|
||||
local bOk = CreateRecursiveSolid( nCrvId, vSurfs, nSolidGrp, LayerParams, dStrand)
|
||||
bOk = bOk and CreateRecursiveSolid( nCrvSplit, vSurfs, nSolidGrp, LayerParams, dStrand)
|
||||
local bOk = CreateRecursiveSolid( nCrvId, vSurfs, nSolidGrp, LayerParams, dH, dStrand)
|
||||
bOk = bOk and CreateRecursiveSolid( nCrvSplit, vSurfs, nSolidGrp, LayerParams, dH, dStrand)
|
||||
return bOk
|
||||
end
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function CreateSolidFromCurve( nCrvId, nSolidGrp, LayerParams, nLayer)
|
||||
local function CalcSolidGuides( nCrvId, dStrand, nSolidGrp)
|
||||
|
||||
-- suddivido la curva in modo opportuno
|
||||
local nCopyId = EgtCopyGlob( nCrvId, nSolidGrp)
|
||||
local nId
|
||||
local nParts = 1
|
||||
local LEN_REF = 100
|
||||
local bZigZagInfill = EgtGetInfo( nCopyId, KEY_ZIG_ZAG_INFILL, 'b') or false
|
||||
if bZigZagInfill then
|
||||
nId, nParts = EgtSplitCurveAtCorners( nCopyId, 80)
|
||||
else
|
||||
local dLen = EgtCurveLength( nCrvId)
|
||||
if dLen > LEN_REF and nType ~= TYPE.LINK and nType ~= TYPE.COASTING then
|
||||
nParts = EgtClamp( floor( dLen / LEN_REF), 1, 10)
|
||||
end
|
||||
nId = EgtSplitCurve( nCopyId, nParts)
|
||||
end
|
||||
|
||||
if not nId then return end
|
||||
|
||||
-- verifico in modo euristico se la curva torna indietro su se stessa come se fossero due shell collegate ( verifico se è presente un sottotratto di lunghezza
|
||||
-- circa pari allo strand non in tangenza con i sottotratti vicini). Nel caso la spezzo per evitare problemi nel calcolo di solidi swept
|
||||
local vGuideIds = {}
|
||||
for nInd = 0, nParts - 1 do
|
||||
local nGuideId = nId + nInd
|
||||
table.insert( vGuideIds, nGuideId)
|
||||
if EgtGetType( nGuideId) == GDB_TY.CRV_COMPO then
|
||||
local _, dE = EgtCurveDomain( nGuideId)
|
||||
for dU = 1, dE - 2 do
|
||||
local dLen = EgtCurveCompoLength( nGuideId, dU)
|
||||
if abs( dLen - dStrand) < 500 * GEO.EPS_SMALL then
|
||||
-- verifico gli angoli
|
||||
local vtPrev = EgtUV( nGuideId, dU, -1, GDB_ID.ROOT)
|
||||
local vtCurrS = EgtUV( nGuideId, dU, 1, GDB_ID.ROOT)
|
||||
local vtCurrE = EgtUV( nGuideId, dU + 1, -1, GDB_ID.ROOT)
|
||||
local vtNext = EgtUV( nGuideId, dU + 1, 1, GDB_ID.ROOT)
|
||||
if vtPrev * vtCurrS < GEO.EPS_SMALL or vtCurrE * vtNext < GEO.EPS_SMALL then
|
||||
local nNewId = EgtSplitCurveAtParam( nGuideId, dU + 0.5)
|
||||
table.insert( vGuideIds, nNewId)
|
||||
break
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
return vGuideIds
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function CreateSolidFromCurve( nCrvId, nSolidGrp, LayerParams, nSliceNbr, dLayerH)
|
||||
|
||||
local nType = EgtGetInfo( nCrvId, KEY_TYPE, 'i')
|
||||
local nType = EgtGetInfo( nCrvId, KEY_TYPE, 'i')
|
||||
if nType == TYPE.WIPE then return true end
|
||||
-- scelta del colore
|
||||
local Color = EgtStdColor( 'GRAY')
|
||||
@@ -292,52 +343,48 @@ local function CreateSolidFromCurve( nCrvId, nSolidGrp, LayerParams, nLayer)
|
||||
elseif nType == TYPE.AUX_SOLID then
|
||||
Color = EgtStdColor( 'AQUA')
|
||||
end
|
||||
local dStrand = EgtGetInfo( nCrvId, KEY_CRV_STRAND, 'd') or LayerParams.dStrand
|
||||
|
||||
local nCopyId = EgtCopyGlob( nCrvId, nSolidGrp)
|
||||
local nId = GDB_ID.NULL
|
||||
local nParts = 1
|
||||
local LEN_REF = 100
|
||||
local bZigZagInfill = EgtGetInfo( nCopyId, KEY_ZIG_ZAG_INFILL, 'b') or false
|
||||
if bZigZagInfill then
|
||||
nId, nParts = EgtSplitCurveAtCorners( nCopyId, 80)
|
||||
else
|
||||
local dLen = EgtCurveLength( nCrvId)
|
||||
if dLen > LEN_REF and nType ~= TYPE.LINK and nType ~= TYPE.COASTING then
|
||||
nParts = EgtClamp( floor( dLen / LEN_REF), 1, 10)
|
||||
end
|
||||
nId = EgtSplitCurve( nCopyId, nParts)
|
||||
if s_nSimplifiedSection == 1 and nSliceNbr % 2 == 0 then
|
||||
Color = Color3d( s_dColorFactor * Color:getRed(), s_dColorFactor * Color:getGreen(), s_dColorFactor * Color:getBlue())
|
||||
end
|
||||
|
||||
-- parametri della passata
|
||||
local dStrand = EgtGetInfo( nCrvId, KEY_CRV_STRAND, 'd') or LayerParams.dStrand
|
||||
local dH = dLayerH or LayerParams.dLayHeight
|
||||
|
||||
-- freccia direzionale
|
||||
local sName = EgtGetName( nCrvId)
|
||||
if nType ~= TYPE.COASTING and nType ~= TYPE.LINK and sName ~= LEAD_IN_CRV and sName ~= LEAD_OUT_CRV and sName ~= LINK_CRV then
|
||||
CreateDirectionArrow( nCrvId, nSolidGrp, LayerParams.vtSlicing, dStrand, nLayer)
|
||||
CreateDirectionArrow( nCrvId, nSolidGrp, LayerParams.vtSlicing, dStrand, nSliceNbr)
|
||||
end
|
||||
|
||||
if nId == GDB_ID.NULL then return false end
|
||||
-- spezzo la curva guida in diversi sottotratti per i quali calcolare i solidi
|
||||
local vGuideIds = CalcSolidGuides( nCrvId, dStrand, nSolidGrp)
|
||||
if not vGuideIds then
|
||||
return false
|
||||
end
|
||||
|
||||
local bOk = true
|
||||
for nInd = 0, nParts - 1 do
|
||||
local nGuideId = nId + nInd
|
||||
local nSrfId = CreateSolid( nGuideId, nSolidGrp, LayerParams, dStrand - 5 * GEO.EPS_SMALL)
|
||||
for i = 1, #vGuideIds do
|
||||
local nSrfId = CreateSolid( vGuideIds[i], nSolidGrp, LayerParams, dH, dStrand - 5 * GEO.EPS_SMALL)
|
||||
if not nSrfId then
|
||||
EgtOutLog( 'Warning : CreateSolid failed '.. '(layer '..tostring( nLayer)..', curve '..tostring( nCrvId)..')')
|
||||
EgtOutLog( 'Warning : CreateSolid failed '.. '(layer '..tostring( nSliceNbr)..', curve '..tostring( nCrvId)..')')
|
||||
-- ritento con strand più piccolo
|
||||
nSrfId = CreateSolid( nGuideId, nSolidGrp, LayerParams, dStrand - 50 * GEO.EPS_SMALL)
|
||||
nSrfId = CreateSolid( vGuideIds[i], nSolidGrp, LayerParams, dH, dStrand - 50 * GEO.EPS_SMALL)
|
||||
|
||||
if not nSrfId then
|
||||
EgtOutLog( 'Warning : CreateSolid_1 failed '.. '(layer '..tostring( nLayer)..', curve '..tostring( nCrvId)..')')
|
||||
EgtOutLog( 'Warning : CreateSolid_1 failed '.. '(layer '..tostring( nSliceNbr)..', curve '..tostring( nCrvId)..')')
|
||||
-- se non ultima, provo a spostare l'estremità finale
|
||||
if nInd < nParts - 1 then
|
||||
local nCopyId = EgtCopy( nGuideId + 1, nGuideId, GDB_IN.AFTER)
|
||||
if i < #vGuideIds then
|
||||
local nCopyId = EgtCopy( vGuideIds[i] + 1, vGuideIds[i], GDB_IN.AFTER)
|
||||
if nCopyId then
|
||||
local LEN_TRIM = 10
|
||||
local bOk1 = EgtTrimCurveEndAtLen( nCopyId, LEN_TRIM)
|
||||
local bOk2 = EgtAddCurveCompoCurve( nGuideId, nCopyId)
|
||||
local bOk3 = EgtTrimCurveStartAtLen( nGuideId + 1, LEN_TRIM)
|
||||
nSrfId = CreateSolid( nGuideId, nSolidGrp, LayerParams, dStrand - 5 * GEO.EPS_SMALL)
|
||||
local bOk2 = EgtAddCurveCompoCurve( vGuideIds[i], nCopyId)
|
||||
local bOk3 = EgtTrimCurveStartAtLen( vGuideIds[i] + 1, LEN_TRIM)
|
||||
nSrfId = CreateSolid( vGuideIds[i], nSolidGrp, LayerParams, dH, dStrand - 5 * GEO.EPS_SMALL)
|
||||
if not nSrfId then
|
||||
nSrfId = CreateSolid( nGuideId, nSolidGrp, LayerParams, dStrand - 50 * GEO.EPS_SMALL)
|
||||
nSrfId = CreateSolid( vGuideIds[i], nSolidGrp, LayerParams, dH, dStrand - 50 * GEO.EPS_SMALL)
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -347,12 +394,12 @@ local function CreateSolidFromCurve( nCrvId, nSolidGrp, LayerParams, nLayer)
|
||||
if not nSrfId then
|
||||
EgtOutLog( 'Warning : CreateSolid_2 failed')
|
||||
local nGrp = EgtGroup( nSolidGrp, Frame3d( ORIG(), LayerParams.vtSlicing), GDB_RT.GLOB)
|
||||
EgtRelocateGlob( nGuideId, nGrp)
|
||||
EgtApproxCurve( nGuideId, GDB_CA.LINES, 100 * GEO.EPS_SMALL)
|
||||
EgtRelocateGlob( nGuideId, nSolidGrp)
|
||||
EgtRelocateGlob( vGuideIds[i], nGrp)
|
||||
EgtApproxCurve( vGuideIds[i], GDB_CA.LINES, 100 * GEO.EPS_SMALL)
|
||||
EgtRelocateGlob( vGuideIds[i], nSolidGrp)
|
||||
|
||||
local vSurfs = {}
|
||||
local bOk = CreateRecursiveSolid( nGuideId, vSurfs, nSolidGrp, LayerParams, dStrand)
|
||||
local bOk = CreateRecursiveSolid( vGuideIds[i], vSurfs, nSolidGrp, LayerParams, dH, dStrand)
|
||||
if #vSurfs > 0 then
|
||||
nSrfId = EgtSurfTmByTriangles( nSolidGrp, vSurfs)
|
||||
end
|
||||
@@ -366,17 +413,381 @@ local function CreateSolidFromCurve( nCrvId, nSolidGrp, LayerParams, nLayer)
|
||||
if nSrfId then
|
||||
EgtSetColor( nSrfId, Color)
|
||||
EgtSetInfo( nSrfId, KEY_TYPE, nType)
|
||||
EgtSetInfo( nSrfId, KEY_SLICE_NBR, nLayer)
|
||||
EgtSetInfo( nSrfId, KEY_SLICE_NBR, nSliceNbr)
|
||||
else
|
||||
bOk = false
|
||||
EgtOutLog( 'Warning : CreateSolid_Sewing failed')
|
||||
end
|
||||
EgtErase( nGuideId)
|
||||
end
|
||||
|
||||
EgtErase( vGuideIds[i])
|
||||
end
|
||||
|
||||
return bOk
|
||||
end
|
||||
|
||||
--------------------------------------------------------------------
|
||||
------------------------- MULTIPLANAR ------------------------------
|
||||
--------------------------------------------------------------------
|
||||
local function CreateMultiPlanarSolids( vIds, nSolidGrpId, LayerParams, nLayerId)
|
||||
|
||||
-- creo solidi standard con spessore massimo ammesso ( tranne nel primo layer che viene fatto ad altezza costante)
|
||||
local nSliceNbr = EgtGetInfo( nLayerId, KEY_SLICE_NBR, 'i')
|
||||
local dH = EgtIf( nSliceNbr == 1, LayerParams.dLayHeight, s_dMultiPlanarH)
|
||||
for i = 1, #vIds do
|
||||
CreateSolidFromCurve( vIds[i], nSolidGrpId, LayerParams, nSliceNbr, dH)
|
||||
end
|
||||
|
||||
-- deformo i solidi per rispettare l'altezza reale della passata
|
||||
if nSliceNbr > 1 then
|
||||
local ptSlicing = EgtGetInfo( nLayerId, KEY_SLICE_POS, 'p') + LayerParams.vtSlicing * LayerParams.dLayHeight
|
||||
local nPrevLayerId = EgtGetPrev( nLayerId)
|
||||
local vtSlicingPrev = EgtGetInfo( nPrevLayerId, KEY_SLICE_DIR, 'v')
|
||||
local ptSlicingPrev = EgtGetInfo( nPrevLayerId, KEY_SLICE_POS, 'p') + vtSlicingPrev * LayerParams.dLayHeight
|
||||
local dCosAng = vtSlicingPrev * LayerParams.vtSlicing
|
||||
|
||||
local vSolids = EgtGetAllInGroup( nSolidGrpId)
|
||||
for i = 1, #vSolids do
|
||||
local nVertexCnt = EgtSurfTmVertexCount( vSolids[i])
|
||||
for j = 0, nVertexCnt - 1 do
|
||||
local ptVertex = EgtSurfTmGetVertex( vSolids[i], j, GDB_RT.GLOB)
|
||||
-- distanza dal piano di slicing corrente e precedente lungo vtSlicing
|
||||
local dDistCurr = abs( ( ptVertex - ptSlicing) * LayerParams.vtSlicing)
|
||||
local dDistPrev = ( ptVertex - ptSlicingPrev) * vtSlicingPrev / dCosAng
|
||||
-- calcolo la nuova posizione sapendo che l'altezza dello strand passa da s_dMultiPlanarH a dNewH
|
||||
local dNewH = dDistCurr + dDistPrev
|
||||
local dDelta = dDistCurr - dDistCurr * dNewH / s_dMultiPlanarH
|
||||
local ptNew = ptVertex + dDelta * LayerParams.vtSlicing
|
||||
EgtSurfTmMoveVertex( vSolids[i], j, ptNew, GDB_RT.GLOB, ( j == nVertexCnt - 1))
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function CreatePartialSpiralVaseMultiPlanarSolids( vIds, nSolidGrpId, LayerParams, nLayerId)
|
||||
|
||||
-- 1) le prime curve sono solidi multiplanari standard
|
||||
for i = 1, #vIds - 1 do
|
||||
CreateMultiPlanarSolids( { vIds[i]}, nSolidGrpId, LayerParams, nLayerId)
|
||||
end
|
||||
|
||||
-- 2) l'ultima curva è il tratto che si alza fino al layer successivo
|
||||
local nGrpTmp = EgtGroup( nSolidGrpId)
|
||||
local dStrand = EgtGetInfo( vIds[#vIds], KEY_CRV_STRAND, 'd') or LayerParams.dStrand
|
||||
|
||||
-- a) calcolo il solido
|
||||
-- appiattisco la curva ( proiezione obliqua sul piano corrente lungo vtSlicing del piano successivo)
|
||||
local nNextLayerId = EgtGetNext( nLayerId)
|
||||
local vtSlicingNext = EgtGetInfo( nNextLayerId, KEY_SLICE_DIR, 'v')
|
||||
local ptSlicing = EgtGetInfo( nLayerId, KEY_SLICE_POS, 'p') + LayerParams.dLayHeight * LayerParams.vtSlicing
|
||||
local nProjCrv = EgtCopyGlob( vIds[#vIds], nGrpTmp)
|
||||
local _, dE = EgtCurveDomain( nProjCrv)
|
||||
for dU = 0, dE do
|
||||
local ptCurr = EgtUP( nProjCrv, dU, GDB_ID.ROOT)
|
||||
local dDist = ( ptCurr - ptSlicing) * LayerParams.vtSlicing / ( vtSlicingNext * LayerParams.vtSlicing)
|
||||
EgtModifyCurveCompoJoint( nProjCrv, dU, ptCurr - dDist * vtSlicingNext, GDB_RT.GLOB)
|
||||
end
|
||||
|
||||
-- calcolo la sezione iniziale
|
||||
local ptS = EgtSP( vIds[#vIds], GDB_ID.ROOT)
|
||||
local ptE = EgtEP( vIds[#vIds], GDB_ID.ROOT)
|
||||
local vtDir = EgtSV( nProjCrv, GDB_ID.ROOT)
|
||||
vtDir:rotate( LayerParams.vtSlicing, 90)
|
||||
local nSectId = CreateSection( ptS, vtDir, dStrand, s_dMultiPlanarH, LayerParams.vtSlicing, nGrpTmp)
|
||||
-- creo la sezione finale
|
||||
local vtDir2 = EgtEV( nProjCrv, GDB_ID.ROOT)
|
||||
vtDir2:rotate( LayerParams.vtSlicing, 90)
|
||||
local nSectE = CreateSection( ptE, vtDir2, dStrand, s_dMultiPlanarH, LayerParams.vtSlicing, nGrpTmp)
|
||||
|
||||
-- creo le guide per il solido aperto ( tubo)
|
||||
local dMaxDist = ( EgtEP( vIds[#vIds], GDB_ID.ROOT) - ptSlicing) * LayerParams.vtSlicing / ( vtSlicingNext * LayerParams.vtSlicing)
|
||||
local vCrvs = {}
|
||||
local _, dParE = EgtCurveDomain( nSectId)
|
||||
for i = 0, dParE do
|
||||
local ptRef = EgtUP( nSectId, i, GDB_ID.ROOT)
|
||||
local dOffs = ( ptS - ptRef) * vtDir
|
||||
vCrvs[i+1] = EgtOffsetCurveAdv( nProjCrv, dOffs)
|
||||
if not vCrvs[i+1] or vCrvs[i+1] == GDB_ID.NULL then
|
||||
EgtErase( nGrpTmp)
|
||||
return false
|
||||
end
|
||||
|
||||
-- proiezione obliqua "graduale"
|
||||
EgtApproxCurve( vCrvs[i+1], GDB_CA.SPECIAL_LINES, 0.01)
|
||||
local _, dE = EgtCurveDomain( vCrvs[i+1])
|
||||
local dLen = EgtCurveLength( vCrvs[i+1])
|
||||
local vDelta = {}
|
||||
for dU = 0, dE do
|
||||
local dCurrLen = EgtCurveLengthAtParam( vCrvs[i+1], dU)
|
||||
vDelta[dU] = dMaxDist * dCurrLen / dLen
|
||||
end
|
||||
for dU = 0, dE do
|
||||
local ptCurr = EgtUP( vCrvs[i+1], dU, GDB_ID.ROOT)
|
||||
EgtModifyCurveCompoJoint( vCrvs[i+1], dU, ptCurr + vDelta[dU] * vtSlicingNext, GDB_RT.GLOB)
|
||||
end
|
||||
|
||||
local dMove = ( ptRef - ptS) * LayerParams.vtSlicing
|
||||
EgtMove( vCrvs[i+1], LayerParams.vtSlicing * dMove, GDB_RT.GLOB)
|
||||
end
|
||||
-- creazione delle superfici del tubo
|
||||
local vSurfs = {}
|
||||
for i = 1, #vCrvs - 1 do
|
||||
vSurfs[i] = EgtSurfTmRuled( nGrpTmp, vCrvs[i], vCrvs[i+1], GDB_RUL.MINDIST, s_dTol)
|
||||
if not vSurfs[i] or vSurfs[i] == GDB_ID.NULL then
|
||||
EgtErase( nGrpTmp)
|
||||
return false
|
||||
end
|
||||
end
|
||||
-- creazione del mezzo disco iniziale e finale
|
||||
vSurfs[#vCrvs] = CreateSpiralVaseCap( nSectId, - EgtSV( vIds[#vIds], GDB_ID.ROOT), nSolidGrpId)
|
||||
vSurfs[#vCrvs + 1] = CreateSpiralVaseCap( nSectE, EgtEV( vIds[#vIds], GDB_ID.ROOT), nSolidGrpId)
|
||||
EgtInvertSurf( vSurfs[#vSurfs])
|
||||
local nSolidId = EgtSurfTmByTriangles( nSolidGrpId, vSurfs)
|
||||
|
||||
-- b) deformo il solido per limitarlo al piano di slicing precedente
|
||||
local ptSlicingPrev
|
||||
local vtSlicingPrev
|
||||
local nSliceNbr = EgtGetInfo( nLayerId, KEY_SLICE_NBR, 'i')
|
||||
if nSliceNbr == 1 then
|
||||
vtSlicingPrev = LayerParams.vtSlicing
|
||||
ptSlicingPrev = ptSlicing - LayerParams.dLayHeight * LayerParams.vtSlicing
|
||||
else
|
||||
local nPrevLayId = EgtGetPrev( nLayerId)
|
||||
vtSlicingPrev = EgtGetInfo( nPrevLayId, KEY_SLICE_DIR, 'v')
|
||||
ptSlicingPrev = EgtGetInfo( nPrevLayId, KEY_SLICE_POS, 'p') + vtSlicingPrev * LayerParams.dLayHeight
|
||||
end
|
||||
local dCosAng = vtSlicingPrev * LayerParams.vtSlicing
|
||||
|
||||
-- estendo la curva proiettata per gestire correttamente la deformazione dei caps
|
||||
local nProjExtCrv = EgtCopyGlob( nProjCrv, nGrpTmp)
|
||||
EgtExtendCurveStartByLen( nProjExtCrv, dStrand)
|
||||
EgtExtendCurveEndByLen( nProjExtCrv, dStrand)
|
||||
local dLen = EgtCurveLength( nProjCrv)
|
||||
|
||||
local nVertexCnt = EgtSurfTmVertexCount( nSolidId)
|
||||
for j = 0, nVertexCnt - 1 do
|
||||
|
||||
local ptVertex = EgtSurfTmGetVertex( nSolidId, j, GDB_RT.GLOB)
|
||||
|
||||
-- calcolo la distanza dal piano precedente lungo vtSlicing
|
||||
local dDistPrevPlane = ( ptVertex - ptSlicingPrev) * vtSlicingPrev / dCosAng
|
||||
-- calcolo la distanza approssimata dalla curva del toolpath
|
||||
local _, ptMinDist, dMinPar = EgtPointCurveDist( ptVertex, nProjExtCrv, GDB_ID.ROOT)
|
||||
local dCurrLen = EgtCurveLengthAtParam( nProjExtCrv, dMinPar) - dStrand
|
||||
local ptRef = ptMinDist + dMaxDist * dCurrLen / dLen * vtSlicingNext
|
||||
local dDistCrv = ( ptRef - ptVertex) * LayerParams.vtSlicing
|
||||
|
||||
-- calcolo la nuova posizione sapendo che l'altezza dello strand passa da s_dMultiPlanarH a dNewH
|
||||
local dNewH = dDistCrv + dDistPrevPlane
|
||||
local dDelta = dDistCrv - dDistCrv * dNewH / s_dMultiPlanarH
|
||||
local ptNew = ptVertex + dDelta * LayerParams.vtSlicing
|
||||
EgtSurfTmMoveVertex( nSolidId, j, ptNew, GDB_RT.GLOB, ( j == nVertexCnt - 1))
|
||||
end
|
||||
|
||||
local Color = EgtStdColor( 'TEAL')
|
||||
if s_nSimplifiedSection == 1 and nSliceNbr % 2 == 0 then
|
||||
Color = Color3d( s_dColorFactor * Color:getRed(), s_dColorFactor * Color:getGreen(), s_dColorFactor * Color:getBlue())
|
||||
end
|
||||
EgtSetColor( nSolidId, Color)
|
||||
|
||||
EgtErase( nGrpTmp)
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function CreateFullSpiralVaseMultiPlanarSolids( vIds, nSolidGrpId, LayerParams, nLayerId)
|
||||
|
||||
local nSliceNbr = EgtGetInfo( nLayerId, KEY_SLICE_NBR, 'i')
|
||||
local ptSlicing = EgtGetInfo( nLayerId, KEY_SLICE_POS, 'p') + LayerParams.dLayHeight * LayerParams.vtSlicing
|
||||
|
||||
-- recupero il piano precedente e il suo toolpath
|
||||
local nPrevLayerId = EgtGetPrev( nLayerId)
|
||||
local vtSlicingPrev = EgtGetInfo( nPrevLayerId, KEY_SLICE_DIR, 'v')
|
||||
local ptSlicingPrev = EgtGetInfo( nPrevLayerId, KEY_SLICE_POS, 'p') + LayerParams.dLayHeight * vtSlicingPrev
|
||||
local dCosAng = vtSlicingPrev * LayerParams.vtSlicing
|
||||
local nPrevCrvGrp = EgtGetFirstNameInGroup( nPrevLayerId, CONTOUR_GRP .. '*')
|
||||
local nPrevTPathGrp = EgtGetFirstNameInGroup( nPrevCrvGrp, TOOLPATH_GRP)
|
||||
local nPrevTPath = EgtGetFirstNameInGroup( nPrevTPathGrp, SHELL_CRV .. '*')
|
||||
|
||||
-- gruppo temporaneo per conti
|
||||
local nGrpTmp = EgtGroup( nSolidGrpId)
|
||||
|
||||
for i = 1, #vIds do
|
||||
local dStrand = EgtGetInfo( vIds[i], KEY_CRV_STRAND, 'd') or LayerParams.dStrand
|
||||
|
||||
-- creo la freccia direzionale
|
||||
CreateDirectionArrow( vIds[i], nSolidGrpId, LayerParams.vtSlicing, dStrand, nSliceNbr)
|
||||
|
||||
-- appiattisco la curva ( proiezione sul piano di slicing)
|
||||
local nProjCrv = EgtCopyGlob( vIds[i], nGrpTmp)
|
||||
EgtProjectCurveOnPlane( nProjCrv, ptSlicing, LayerParams.vtSlicing, GDB_RT.GLOB)
|
||||
local dLen = EgtCurveLength( nProjCrv)
|
||||
|
||||
-- suddivido la curva piana
|
||||
local vGuideIds = CalcSolidGuides( nProjCrv, dStrand, nGrpTmp)
|
||||
local dCumLen = 0
|
||||
for k = 1, #vGuideIds do
|
||||
|
||||
-- calcolo il solido sulla porzione della curva di proiezione
|
||||
local nSolidId = CreateStandardSolid( vGuideIds[k], nSolidGrpId, s_dMultiPlanarH, dStrand)
|
||||
|
||||
-- estendo la guida per gestire al meglio i caps
|
||||
local nCrvRef = EgtCopyGlob( vGuideIds[k], nGrpTmp)
|
||||
EgtExtendCurveStartByLen( nCrvRef, dStrand)
|
||||
EgtExtendCurveEndByLen( nCrvRef, dStrand)
|
||||
|
||||
-- recupero la curva precedente. Se estremi la limito per gestire al meglio la corrispondenza tra i toolpath
|
||||
local nPrevTPathRef = nPrevTPath
|
||||
if k == 1 then
|
||||
nPrevTPathRef = EgtCopyGlob( nPrevTPath, nGrpTmp)
|
||||
EgtTrimCurveEndAtLen( nPrevTPathRef, 0.5 * EgtCurveLength( nPrevTPathRef))
|
||||
elseif k == #vGuideIds then
|
||||
nPrevTPathRef = EgtCopyGlob( nPrevTPath, nGrpTmp)
|
||||
EgtTrimCurveStartAtLen( nPrevTPathRef, 0.5 * EgtCurveLength( nPrevTPathRef))
|
||||
end
|
||||
|
||||
-- deformo i solidi :
|
||||
-- 1) traslazione dei vertici per portarli alla quota corretta del toolpath corrente
|
||||
-- 2) deformazione della sezione in base alla distanza dal toolpath precedente
|
||||
local nVertexCnt = EgtSurfTmVertexCount( nSolidId)
|
||||
for j = 0, nVertexCnt - 1 do
|
||||
local ptVertex = EgtSurfTmGetVertex( nSolidId, j, GDB_RT.GLOB)
|
||||
|
||||
-- ricavo il punto di riferimento sulla proiezione del toolpath
|
||||
local _, ptMinDist, dParMinDist = EgtPointCurveDist( ptVertex, nCrvRef, GDB_ID.ROOT)
|
||||
local dCurrLen = EgtCurveLengthAtParam( nCrvRef, dParMinDist) - dStrand + dCumLen
|
||||
|
||||
-- 1) proietto il punto sul piano corrente e calcolo la traslazione da applicare in base alla distanza dal piano precedente ( analogamente a CalcToolPath)
|
||||
local dDistCrv = ( ptSlicing - ptVertex) * LayerParams.vtSlicing
|
||||
local ptProj = ptVertex + dDistCrv * LayerParams.vtSlicing
|
||||
local dDistPlanePrev = ( ptProj - ptSlicingPrev) * vtSlicingPrev / dCosAng
|
||||
local dDeltaPos
|
||||
if nSliceNbr == 2 and dCurrLen < 0.5 * dLen then
|
||||
dDeltaPos = 0.5 * dDistPlanePrev
|
||||
else
|
||||
dDeltaPos = dDistPlanePrev * ( dLen - dCurrLen) / dLen
|
||||
end
|
||||
|
||||
-- 2) per calcolare la vera altezza del solido calcolo la distanza dal toolpath precedente guardando il punto di riferimento sul percorso corrente
|
||||
local dNewH
|
||||
if nSliceNbr == 2 then
|
||||
-- il primo layer è ad altezza costante quindi è semplice distanza dal piano precedente lungo vtSlicing
|
||||
dNewH = ( ptProj - ptSlicingPrev) * vtSlicingPrev / dCosAng - dDeltaPos
|
||||
else
|
||||
local _, ptMinDistPrev = EgtPointCurveDist( ptMinDist, nPrevTPathRef, GDB_ID.ROOT)
|
||||
local dDistPlanePrev2 = ( ptMinDist - ptSlicingPrev) * vtSlicingPrev / dCosAng
|
||||
local dDeltaPos2
|
||||
if nSliceNbr == 2 and dCurrLen < 0.5 * dLen then
|
||||
dDeltaPos2 = 0.5 * dDistPlanePrev2
|
||||
else
|
||||
dDeltaPos2 = dDistPlanePrev2 * ( dLen - dCurrLen) / dLen
|
||||
end
|
||||
dNewH = ( ptMinDist - ptMinDistPrev) * LayerParams.vtSlicing - dDeltaPos2
|
||||
-- aggiungo un piccolo extra per tener conto che la distanza è stata calcolata sulle curve centrali del solido
|
||||
local dExtra = ( LayerParams.vtSlicing ^ vtSlicingPrev):len() / dCosAng * dStrand * 0.5
|
||||
dNewH = dNewH + dExtra
|
||||
end
|
||||
|
||||
local dDelta = dDistCrv - dDistCrv * dNewH / s_dMultiPlanarH
|
||||
local ptNew = ptVertex + ( dDelta - dDeltaPos) * LayerParams.vtSlicing
|
||||
EgtSurfTmMoveVertex( nSolidId, j, ptNew, GDB_RT.GLOB, ( j == nVertexCnt - 1))
|
||||
end
|
||||
|
||||
-- imposto colore
|
||||
local Color = EgtStdColor( 'TEAL')
|
||||
if s_nSimplifiedSection == 1 and nSliceNbr % 2 == 0 then
|
||||
Color = Color3d( s_dColorFactor * Color:getRed(), s_dColorFactor * Color:getGreen(), s_dColorFactor * Color:getBlue())
|
||||
end
|
||||
EgtSetColor( nSolidId, Color)
|
||||
|
||||
dCumLen = dCumLen + EgtCurveLength( vGuideIds[i])
|
||||
end
|
||||
end
|
||||
|
||||
EgtErase( nGrpTmp)
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
local function CreateFullSpiralVaseMultiPlanarSolidsLastLayer( vIds, nSolidGrpId, LayerParams, nLayerId)
|
||||
-- le curve sono piane quindi i solidi possono essere calcolati nel modo classico
|
||||
-- i solidi della shell sono deformati in base alla distanza dalla curva precedente
|
||||
-- eventuali solidi di lead out e coasting sono calcolati con un valore di altezza costante pari all'ultimo valore di altezza dei solidi della shell
|
||||
|
||||
local nSliceNbr = EgtGetInfo( nLayerId, KEY_SLICE_NBR, 'i')
|
||||
|
||||
local nPrevLayerId = EgtGetPrev( EgtGetPrev( nLayerId))
|
||||
local vtSlicingPrev = EgtGetInfo( nPrevLayerId, KEY_SLICE_DIR, 'v')
|
||||
local ptSlicingPrev = EgtGetInfo( nPrevLayerId, KEY_SLICE_POS, 'p') + vtSlicingPrev * LayerParams.dLayHeight
|
||||
local ptSlicing = EgtGetInfo( nLayerId, KEY_SLICE_POS, 'p') + LayerParams.vtSlicing * LayerParams.dLayHeight
|
||||
local dCosAng = vtSlicingPrev * LayerParams.vtSlicing
|
||||
|
||||
-- 1) curva shell
|
||||
local nGrpTmp = EgtGroup( nSolidGrpId)
|
||||
local dStrand = EgtGetInfo( vIds[1], KEY_CRV_STRAND, 'd') or LayerParams.dStrand
|
||||
local vShellGuides = CalcSolidGuides( vIds[1], dStrand, nGrpTmp)
|
||||
|
||||
local dLen = EgtCurveLength( vIds[1])
|
||||
-- se non c'è lead out alla lunghezza totale della shell devo aggiungere anche quella del coasting
|
||||
if EgtGetInfo( s_nPartId, KEY_LEAD_OUT_TYPE, 'i') == LEAD_TYPE.NONE then
|
||||
dLen = dLen + EgtGetInfo( s_nPartId, KEY_COASTING_LEN, 'd')
|
||||
end
|
||||
local dCumLen = 0
|
||||
|
||||
-- creo la freccia direzionale
|
||||
CreateDirectionArrow( vIds[1], nSolidGrpId, LayerParams.vtSlicing, dStrand, nSliceNbr)
|
||||
|
||||
for i = 1, #vShellGuides do
|
||||
|
||||
-- calcolo il solido associato
|
||||
local nSolidId = CreateStandardSolid( vShellGuides[i], nSolidGrpId, s_dMultiPlanarH, dStrand)
|
||||
|
||||
-- estensione della guida per gestire meglio i caps
|
||||
local nCrvRef = EgtCopyGlob( vShellGuides[i], nGrpTmp)
|
||||
EgtExtendCurveStartByLen( nCrvRef, dStrand)
|
||||
EgtExtendCurveEndByLen( nCrvRef, dStrand)
|
||||
|
||||
-- deformo in base alla distanza dalla curva precedente
|
||||
local nVertexCnt = EgtSurfTmVertexCount( nSolidId)
|
||||
for j = 0, nVertexCnt - 1 do
|
||||
local ptVertex = EgtSurfTmGetVertex( nSolidId, j, GDB_RT.GLOB)
|
||||
|
||||
-- ricavo il punto di riferimento sulla guida
|
||||
local _, _, dParMinDist = EgtPointCurveDist( ptVertex, nCrvRef, GDB_ID.ROOT)
|
||||
local dCurrLen = EgtCurveLengthAtParam( nCrvRef, dParMinDist) - dStrand + dCumLen
|
||||
|
||||
-- calcolo la nuova altezza del solido in base alla distanza dalla curva precedente
|
||||
local dDistCurr = ( ptSlicing - ptVertex) * LayerParams.vtSlicing
|
||||
local ptProj = ptVertex + dDistCurr * LayerParams.vtSlicing
|
||||
local dDistPrev = ( ptProj - ptSlicingPrev) * vtSlicingPrev / dCosAng
|
||||
local dNewH = dDistPrev * ( dLen - dCurrLen) / dLen
|
||||
dNewH = max( dNewH, 500 * GEO.EPS_SMALL)
|
||||
|
||||
-- calcolo la nuova posizione sapendo che l'altezza dello strand passa da s_dMultiPlanarH a dNewH
|
||||
local dDelta = dDistCurr - dDistCurr * dNewH / s_dMultiPlanarH
|
||||
local ptNew = ptVertex + dDelta * LayerParams.vtSlicing
|
||||
EgtSurfTmMoveVertex( nSolidId, j, ptNew, GDB_RT.GLOB, ( j == nVertexCnt - 1))
|
||||
end
|
||||
|
||||
dCumLen = dCumLen + EgtCurveLength( vShellGuides[i])
|
||||
|
||||
-- imposto colore
|
||||
local Color = EgtStdColor( 'TEAL')
|
||||
if s_nSimplifiedSection == 1 and nSliceNbr % 2 == 0 then
|
||||
Color = Color3d( s_dColorFactor * Color:getRed(), s_dColorFactor * Color:getGreen(), s_dColorFactor * Color:getBlue())
|
||||
end
|
||||
EgtSetColor( nSolidId, Color)
|
||||
end
|
||||
|
||||
-- 2) curve di uscita
|
||||
-- calcolo altezza finale del solido shell
|
||||
local ptE = EgtEP( vIds[1], GDB_ID.ROOT)
|
||||
local dH = ( ( ptE - ptSlicingPrev) * vtSlicingPrev / dCosAng) * ( dLen - EgtCurveLength( vIds[1])) / dLen
|
||||
dH = max( dH, 500 * GEO.EPS_SMALL)
|
||||
for i = 2, #vIds do
|
||||
CreateSolidFromCurve( vIds[i], nSolidGrpId, LayerParams, nSliceNbr, dH)
|
||||
end
|
||||
|
||||
EgtErase( nGrpTmp)
|
||||
end
|
||||
|
||||
---------------------------------------------------------------------
|
||||
function RunCalcSolids.Exec()
|
||||
|
||||
@@ -391,6 +802,7 @@ function RunCalcSolids.Exec()
|
||||
local nPartId = EgtGetFirstNameInGroup( GDB_ID.ROOT, PART .. nPartIndex) or EgtGetFirstNameInGroup( GDB_ID.ROOT, PART)
|
||||
while nPartId do
|
||||
|
||||
s_nPartId = nPartId
|
||||
if EgtGetInfo( nPartId, KEY_PART_ON_TABLE, 'b') then
|
||||
-- verifico se necessario calcolare il solido
|
||||
local bCalcSolid = EgtGetInfo( nPartId, KEY_CALC_SOLIDS, 'b') or false
|
||||
@@ -410,15 +822,24 @@ function RunCalcSolids.Exec()
|
||||
end
|
||||
|
||||
EgtSetInfo( nPartId, KEY_HAS_SOLIDS, 1)
|
||||
local nSlicingType = EgtGetInfo( nPartId, KEY_SLICING_TYPE, 'i')
|
||||
|
||||
-- recupero i parametri necessari al calcolo dei solidi
|
||||
local LayerParams = GetLayerParamsForSolidCalc( nPartId)
|
||||
|
||||
-- se slicing multiplanare considero come altezza standard quella massima ammessa
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
local sMachIni = EgtGetCurrMachineDir() .. '\\' .. EgtGetCurrMachineName() .. '.ini'
|
||||
local dMaxFactor = EgtGetNumberFromIni( '3dPrinting', KEY_MAX_STRANDH_FACTOR, 2, sMachIni)
|
||||
s_dMultiPlanarH = min( dMaxFactor, 5) * LayerParams.dLayHeight
|
||||
end
|
||||
|
||||
for nIdx = 1, #vLayIds do
|
||||
|
||||
-- flag di interruzione perchè trovati solidi già ok
|
||||
local bSolidsOk = false
|
||||
-- indice layer (per log)
|
||||
local nLayer = EgtGetInfo( vLayIds[ nIdx], KEY_SLICE_NBR, 'i')
|
||||
-- indice layer ( per log)
|
||||
local nSliceNbr = EgtGetInfo( vLayIds[ nIdx], KEY_SLICE_NBR, 'i')
|
||||
-- scorro tutti i gruppi di contorni
|
||||
local nCrvGrpId = EgtGetFirstNameInGroup( vLayIds[ nIdx], CONTOUR_GRP.."*") or GDB_ID.NULL
|
||||
while nCrvGrpId ~= GDB_ID.NULL do
|
||||
@@ -436,10 +857,43 @@ function RunCalcSolids.Exec()
|
||||
nSolidGrpId = EgtGroup( nCrvGrpId)
|
||||
EgtSetName( nSolidGrpId, SOLID_GRP)
|
||||
EgtSetLevel( nSolidGrpId, GDB_LV.TEMP)
|
||||
|
||||
local vIds = EgtGetAllInGroup( nTPathGrpId)
|
||||
|
||||
-- spiral vase
|
||||
if LayerParams.bSpiralVase then
|
||||
-- slicing multiplanare
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
||||
LayerParams.vtSlicing = EgtGetInfo( vLayIds[nIdx], KEY_SLICE_DIR, 'v')
|
||||
|
||||
if LayerParams.bSpiralVase then
|
||||
local dTransitionLen = EgtGetInfo( nPartId, KEY_SPIRAL_VASE_LEN, 'd') or 0
|
||||
-- a) spiral vase completo ( con transizione in altezza su tutto il layer)
|
||||
if dTransitionLen < GEO.EPS_SMALL then
|
||||
if nIdx == 1 then
|
||||
-- il primo layer va gestito come uno spiral vase con transizione parziale ( curva a quota costante e ultimo tratto che si alza)
|
||||
CreatePartialSpiralVaseMultiPlanarSolids( vIds, nSolidGrpId, LayerParams, vLayIds[nIdx])
|
||||
elseif nIdx == #vLayIds then
|
||||
-- gestione speciale per lead out e coasting
|
||||
CreateFullSpiralVaseMultiPlanarSolidsLastLayer( vIds, nSolidGrpId, LayerParams, vLayIds[nIdx])
|
||||
else
|
||||
CreateFullSpiralVaseMultiPlanarSolids( vIds, nSolidGrpId, LayerParams, vLayIds[nIdx])
|
||||
end
|
||||
|
||||
-- b) spiral vase con transizione parziale
|
||||
else
|
||||
if nIdx == #vLayIds then
|
||||
-- ultimo layer è un multiplanare standard perchè non ha tratto finale che si alza
|
||||
CreateMultiPlanarSolids( vIds, nSolidGrpId, LayerParams, vLayIds[nIdx])
|
||||
else
|
||||
CreatePartialSpiralVaseMultiPlanarSolids( vIds, nSolidGrpId, LayerParams, vLayIds[nIdx])
|
||||
end
|
||||
end
|
||||
-- c) multiplanare standard
|
||||
else
|
||||
CreateMultiPlanarSolids( vIds, nSolidGrpId, LayerParams, vLayIds[nIdx])
|
||||
end
|
||||
|
||||
-- slicing standard spiral vase
|
||||
elseif LayerParams.bSpiralVase then
|
||||
-- i tratti di ingresso, uscita e quelli a quota costante vanno gestiti singolarmente, quelli a quota variabile vanno concatenati
|
||||
local vChainedIds = {}
|
||||
local vTmpIds = {}
|
||||
@@ -465,21 +919,21 @@ function RunCalcSolids.Exec()
|
||||
|
||||
for i = 1, #vChainedIds do
|
||||
if #vChainedIds[i] == 1 then
|
||||
CreateSolidFromCurve( vChainedIds[i][1], nSolidGrpId, LayerParams, nLayer)
|
||||
CreateSolidFromCurve( vChainedIds[i][1], nSolidGrpId, LayerParams, nSliceNbr)
|
||||
else
|
||||
local nNewCrv = EgtCurveCompo( nSolidGrpId, vChainedIds[i], false)
|
||||
local dStrand = EgtGetInfo( vChainedIds[i][1], KEY_CRV_STRAND, 'd') or LayerParams.dStrand
|
||||
EgtSetInfo( nNewCrv, KEY_CRV_STRAND, dStrand)
|
||||
EgtSetInfo( nNewCrv, KEY_TYPE, TYPE.OUTER_SHELL)
|
||||
CreateSolidFromCurve( nNewCrv, nSolidGrpId, LayerParams, nLayer)
|
||||
CreateSolidFromCurve( nNewCrv, nSolidGrpId, LayerParams, nSliceNbr)
|
||||
EgtErase( nNewCrv)
|
||||
end
|
||||
end
|
||||
|
||||
-- caso standard
|
||||
-- slicing standard
|
||||
else
|
||||
for i = 1, #vIds do
|
||||
CreateSolidFromCurve( vIds[i], nSolidGrpId, LayerParams, nLayer)
|
||||
CreateSolidFromCurve( vIds[i], nSolidGrpId, LayerParams, nSliceNbr)
|
||||
end
|
||||
end
|
||||
else
|
||||
|
||||
@@ -33,6 +33,7 @@ function RunGcodeGenerate.Exec()
|
||||
EgtOutBox( 'Error missing part', 'GcodeGenerate')
|
||||
return
|
||||
end
|
||||
local nSlicingType = EgtGetInfo( nPartId, KEY_SLICING_TYPE, 'i')
|
||||
|
||||
-- Recupero i layer da processare
|
||||
local vLayIds = EgtGetNameInGroup( nPartId, SLICE_LAYER.."*")
|
||||
@@ -89,8 +90,15 @@ function RunGcodeGenerate.Exec()
|
||||
-- Rimuovo eventuali precedenti lavorazioni
|
||||
EgtRemoveAllOperations()
|
||||
|
||||
-- Determino lavorazione di libreria
|
||||
local sExtrName = 'Extrusion'
|
||||
if nSlicingType == SLICING_TYPE.HORIZONTAL then
|
||||
local sMachIni = EgtGetCurrMachineDir() .. '\\' .. EgtGetCurrMachineName() .. '.ini'
|
||||
sExtrName = EgtGetStringFromIni( SEC_3DPRINTING, KEY_HORIZ_EXTR, sExtrName, sMachIni)
|
||||
end
|
||||
|
||||
-- Aggiungo la lavorazione
|
||||
local nMchId = EgtAddMachining( 'Extrusion 1', 'Extrusion')
|
||||
local nMchId = EgtAddMachining( 'Extrusion 1', sExtrName)
|
||||
if not nMchId then
|
||||
EgtOutBox( 'Error adding Extrusion', 'GcodeGenerate')
|
||||
return
|
||||
|
||||
@@ -100,7 +100,7 @@ local function CalcCurves( nSliceId, nDestGrp, vtSlicing, dStrandBase)
|
||||
end
|
||||
|
||||
------------------------------------------------------------------
|
||||
local function CalcSpiralVase( nSliceId, nDestGrp)
|
||||
local function GetSolids( nSliceId, nDestGrp)
|
||||
|
||||
-- scorro tutti i gruppi di curve
|
||||
local vCrvs = EgtGetNameInGroup( nSliceId, CONTOUR_GRP .. '*')
|
||||
@@ -156,10 +156,11 @@ function RunPrepareExport.Exec()
|
||||
-- recupero alcune info utili e le assegno al punto per poterle leggere da ThreeJS
|
||||
local dH = EgtGetInfo( nPartId, KEY_SLICE_STEP, 'd')
|
||||
EgtSetInfo( nPnt, KEY_SLICE_STEP, dH)
|
||||
local vtSlicing = EgtGetInfo( nPartId, KEY_SLICING_DIR, 'v')
|
||||
EgtSetInfo( nPnt, KEY_SLICING_DIR, vtSlicing)
|
||||
local vtSlicing = EgtGetInfo( nPartId, KEY_SLICE_DIR, 'v') or EgtGetInfo( nPartId, "SlicingDir", 'v')
|
||||
EgtSetInfo( nPnt, "SlicingDir", vtSlicing)
|
||||
local bSpiralVase = EgtGetInfo( nPartId, KEY_SPIRAL_VASE, 'b') or false
|
||||
EgtSetInfo( nPnt, KEY_SPIRAL_VASE, bSpiralVase)
|
||||
local nSlicingType = EgtGetInfo( nPartId, KEY_SLICING_TYPE, 'i')
|
||||
|
||||
-- recupero lo strand dai parametri generale nel caso non fosse definito sulle singole curve
|
||||
local dStrandBase = EgtGetInfo( nPartId, KEY_STRAND, 'd') or 0
|
||||
@@ -195,8 +196,9 @@ function RunPrepareExport.Exec()
|
||||
EgtSetName( nDestGrp, EgtGetName( vSlices[i]))
|
||||
EgtSetStatus( nDestGrp, GDB_ST.OFF)
|
||||
|
||||
if bSpiralVase then
|
||||
CalcSpiralVase( vSlices[i], nDestGrp)
|
||||
-- se multiplanare o spiral vase esporto direttamente i solidi altrimenti esporto le curve che verranno estruse nel visualizzatore
|
||||
if nSlicingType == SLICING_TYPE.MULTIPLANAR or bSpiralVase then
|
||||
GetSolids( vSlices[i], nDestGrp)
|
||||
else
|
||||
CalcCurves( vSlices[i], nDestGrp, vtSlicing, dStrandBase)
|
||||
end
|
||||
|
||||
+2
-2
@@ -1,4 +1,4 @@
|
||||
-- Version.lua by Egaltech s.r.l. 2025/09/12
|
||||
-- Version.lua by Egaltech s.r.l. 2026/03/23
|
||||
-- Gestione della versione di 3dPrinting
|
||||
|
||||
VERSION = '3.1a1'
|
||||
VERSION = '3.1c3'
|
||||
Reference in New Issue
Block a user