2d78f4d348
- ottimizzazioni nella fase di slicing - migliorata la funzione che chiude buchi nei setti.
1508 lines
58 KiB
Lua
1508 lines
58 KiB
Lua
-- CalcSlices.lua by Egaltech s.r.l. 2023/08/28
|
|
-- Calcolo percorsi di lavoro per Stampa 3d
|
|
|
|
-- Tabella per definizione modulo
|
|
local CalcSlices = {}
|
|
|
|
-- Intestazioni
|
|
require( 'EgtBase')
|
|
|
|
EgtOutLog( ' CalcSlices started', 1)
|
|
|
|
-- Dati
|
|
local AMD = require( 'AddManData')
|
|
|
|
---------------------------------------------------------------------
|
|
local s_nPartId
|
|
local s_vErr = {}
|
|
local s_dStrand
|
|
|
|
-- costanti
|
|
-- 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 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
|
|
|
|
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 vZSlices = { dZmin + DELTAZ}
|
|
local dPosZ = dZmin + dSliceStep
|
|
while dPosZ < dZmax and nCnt < nMaxSlicesNbr do
|
|
table.insert( vZSlices, dPosZ)
|
|
dPosZ = dPosZ + dSliceStep
|
|
nCnt = nCnt + 1
|
|
end
|
|
return vZSlices, vSlicingDir
|
|
end
|
|
|
|
---------------------------------------------------------------------
|
|
local function ComputeMultiplanarSlicingData( dSliceStep, nMaxSlicesNbr, nSlicingType)
|
|
-- 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
|
|
|
|
if nSlicingType == SLICING_TYPE.MULTIPLANAR then
|
|
-- 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
|
|
end
|
|
|
|
local vSlicingDir = {}
|
|
local vSlicingPnt = {}
|
|
if nSlicingType == SLICING_TYPE.MULTIPLANAR_DEG45 then
|
|
dSliceStep = dSliceStep / ( sqrt( 2) / 2)
|
|
end
|
|
|
|
-- ciclo sulla curva
|
|
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)
|
|
local vtTang = EgtUV( nSpineId, dU, -1, GDB_ID.ROOT)
|
|
if nSlicingType == SLICING_TYPE.MULTIPLANAR_DEG45 then
|
|
local vtOrth = VectorFromRotated( vtTang, Z_AX(), -90)
|
|
vtTang:rotate( vtOrth, 45)
|
|
end
|
|
vSlicingDir[i+1] = vtTang
|
|
vSlicingPnt[i+1] = EgtUP( nSpineId, dU, GDB_ID.ROOT)
|
|
end
|
|
|
|
return vSlicingPnt, vSlicingDir
|
|
end
|
|
|
|
--------------------------------------------------------------------
|
|
local function ComputeMaxH( vIds, vtSlicing, dSliceStep)
|
|
|
|
local HMax = EgtGetInfo( s_nPartId, KEY_MAX_H, 'd') or GEO.INFINITO
|
|
if HMax < GEO.EPS_SMALL then
|
|
return GEO.INFINITO
|
|
end
|
|
|
|
-- calcolo il box globale
|
|
local b3BoxGlob = BBox3d()
|
|
for i = 1, #vIds do
|
|
local b3Box = EgtGetBBoxGlob( vIds[i], GDB_BB.STANDARD)
|
|
b3BoxGlob:Add( b3Box)
|
|
end
|
|
EgtSetInfo( s_nPartId, KEY_BOX_MIN_Z, b3BoxGlob:getMin():getZ())
|
|
|
|
-- 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( vtSlicing, Z_AX()) then
|
|
-- se slicing verticale o multiplanare l'altezza è quella letta
|
|
return HMax
|
|
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)
|
|
end
|
|
local b3Box = EgtGetBBoxGlob( nGrpTmp, GDB_BB.STANDARD)
|
|
EgtErase( nGrpTmp)
|
|
if not b3Box then
|
|
return GEO.INFINITO
|
|
end
|
|
local ptMin = b3Box:getMin()
|
|
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 * vtSlicing
|
|
ptMax = ptMax - dSliceStep * vtSlicing
|
|
local frSlicing = Frame3d( ORIG(), vtSlicing)
|
|
ptMin:toLoc( frSlicing)
|
|
ptMax:toLoc( frSlicing)
|
|
|
|
return min( ptMin:getZ(), ptMax:getZ())
|
|
end
|
|
|
|
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)
|
|
-- altrimenti recupero info dai parametri generali
|
|
if info == nil then info = EgtGetInfo( s_nPartId, sKey, sType) end
|
|
-- o assegno valore di default
|
|
if info == nil then info = defVal end
|
|
|
|
-- se presente, inserisco l'info nella tabella
|
|
if table then table[sKey] = info end
|
|
|
|
return info
|
|
end
|
|
|
|
-----------------------------------------------------------------------
|
|
local function GetRibParams( nId)
|
|
|
|
local RibParam = {}
|
|
ReadParam( nId, KEY_RIBS_STRAND, 'd', s_dStrand, RibParam)
|
|
if RibParam[KEY_RIBS_STRAND] < GEO.EPS_SMALL then
|
|
RibParam[KEY_RIBS_STRAND] = s_dStrand
|
|
end
|
|
ReadParam( nId, KEY_RIBS_SHELLS_NBR, 'i', 0, RibParam)
|
|
ReadParam( nId, KEY_RIBS_INVERT_DIR, 'b', false, RibParam)
|
|
ReadParam( nId, KEY_RIBS_LEAD_IN_INVERT, 'b', false, RibParam)
|
|
ReadParam( nId, KEY_RIBS_LEAD_IN_LEN, 'd', 0, RibParam)
|
|
ReadParam( nId, KEY_RIBS_LEAD_OUT_INVERT, 'b', false, RibParam)
|
|
ReadParam( nId, KEY_RIBS_LEAD_OUT_LEN, 'd', 0, RibParam)
|
|
ReadParam( nId, KEY_RIBS_LEAD_OUT_COASTING, 'd', 0, RibParam)
|
|
ReadParam( nId, KEY_RIBS_LEAD_OUT_WIPE, 'd', 0, RibParam)
|
|
ReadParam( nId, KEY_RIBS_LEAD_OUT_WIPE_DIR, 'd', 0, RibParam)
|
|
ReadParam( nId, KEY_RIBS_OVERLAP, 'd', 0, RibParam)
|
|
ReadParam( nId, KEY_RIBS_STRAND_OVERLAP, 'd', 0, RibParam)
|
|
ReadParam( nId, KEY_RIBS_LINK, 'b', false, RibParam)
|
|
ReadParam( nId, KEY_RIBS_TYPE, 'i', RIB_TYPE.INTERNAL, RibParam)
|
|
ReadParam( nId, KEY_RIBS_INVERT_STRAND_ORDER, 'b', false, RibParam)
|
|
ReadParam( nId, KEY_RIBS_MERGE_WITH_SHELLS, 'b', false, RibParam)
|
|
ReadParam( nId, KEY_RIBS_LEAD_FILLET, 'd', 0, RibParam)
|
|
ReadParam( nId, KEY_RIBS_LINK_FILLET, 'd', 0, RibParam)
|
|
|
|
return RibParam
|
|
end
|
|
|
|
-------------------------------------------------------------------
|
|
local function GetShellNumberParams( nId)
|
|
|
|
local ShellNumberParam = {}
|
|
ReadParam( nId, KEY_SHELL_NBR_DIFF, 'i', 0, ShellNumberParam)
|
|
ReadParam( nId, KEY_SHELL_NBR_COASTING, 'd', 0, ShellNumberParam)
|
|
ReadParam( nId, KEY_SHELL_NBR_WIPE, 'd', 0, ShellNumberParam)
|
|
ReadParam( nId, KEY_SHELL_NBR_WIPE_DIR, 'd', 0, ShellNumberParam)
|
|
|
|
return ShellNumberParam
|
|
end
|
|
|
|
--------------------------------------------------------------------
|
|
local function GetAuxSolidsParams( nId)
|
|
|
|
local AuxSolidsParam = {}
|
|
ReadParam( nId, KEY_AUX_SOLIDS_STRAND, 'd', s_dStrand, AuxSolidsParam)
|
|
if AuxSolidsParam[KEY_AUX_SOLIDS_STRAND] < GEO.EPS_SMALL then
|
|
AuxSolidsParam[KEY_AUX_SOLIDS_STRAND] = s_dStrand
|
|
end
|
|
ReadParam( nId, KEY_AUX_SOLIDS_SHELLS_NBR, 'i', 0, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_OVERLAP, 'd', 0, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_INFILL, 'i', FILL_TYPE.NONE, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_PRINT_ORDER, 'i', PRINT_ORDER.EXT_INT, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_LINK_TYPE, 'i', LINK_TYPE.NONE, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_LINK_PARAM, 'd', 0, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_SP_OFFSET, 'd', 0, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_LP_OFFSET, 'd', 0, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_DENSITY, 'd', 0, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_DIR, 'd', 0, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_INFILL_LINK, 'b', false, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_OFFSET_X, 'd', 0, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_OFFSET_Y, 'd', 0, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_GRID_OVERLAP, 'd', 0, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_COASTING_LEN, 'd', 0, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_WIPE_LEN, 'd', 0, AuxSolidsParam)
|
|
ReadParam( nId, KEY_AUX_SOLIDS_WIPE_DIR, 'd', 0, AuxSolidsParam)
|
|
|
|
return AuxSolidsParam
|
|
end
|
|
|
|
--------------------------------------------------------------------
|
|
local function SlicingExtraObjects( nLay, vSlicingDir, vSlicingVal, nSlicingType, nType, sName, sNameGrp, nStmId)
|
|
|
|
if not nLay then return end
|
|
|
|
-- recupero gli oggetti di cui fare slicing
|
|
local vIds = EgtGetAllInGroup( nLay)
|
|
if not vIds or #vIds == 0 then return end
|
|
|
|
-- recupero i parametri di ogni oggetto e verifico se da considerare per lo slicing
|
|
local tabParams = {}
|
|
local vSliceIds = {}
|
|
for i = 1, #vIds do
|
|
-- verifico se trimesh
|
|
if EgtGetType( vIds[i]) == GDB_TY.SRF_MESH then
|
|
local bToBeDone = true
|
|
-- recupero i parametri
|
|
local vParams = {}
|
|
if nType == TYPE.RIB then
|
|
vParams = GetRibParams( vIds[i])
|
|
if vParams[KEY_RIBS_SHELLS_NBR] == 0 then bToBeDone = false end
|
|
elseif nType == TYPE.EXTRA_SHELL then
|
|
vParams = GetShellNumberParams( vIds[i])
|
|
if vParams[KEY_SHELL_NBR_DIFF] == 0 then bToBeDone = false end
|
|
elseif nType == TYPE.AUX_SOLID then
|
|
vParams = GetAuxSolidsParams( vIds[i])
|
|
end
|
|
-- se da fare lo aggiungo nei vettori
|
|
if bToBeDone then
|
|
table.insert( tabParams, vParams)
|
|
table.insert( vSliceIds, vIds[i])
|
|
end
|
|
end
|
|
end
|
|
|
|
-- taglio i setti unbounded e limitati con il solido
|
|
local vEraseIds = {}
|
|
if nStmId then
|
|
local bLimitUnbddRibs = EgtGetInfo( s_nPartId, KEY_LIMIT_UNBDD_RIBS, 'b') or false
|
|
if nType == TYPE.RIB and bLimitUnbddRibs then
|
|
for i = 1, #vSliceIds do
|
|
if tabParams[i][KEY_RIBS_TYPE] == RIB_TYPE.UNBOUNDED then
|
|
-- creo copia e la taglio con solido
|
|
local nCopy = EgtCopyGlob( vSliceIds[i], EgtGetParent( vSliceIds[i]))
|
|
EgtSurfTmCut( nCopy, nStmId, true, true)
|
|
-- lo slicing va fatto sulla copia
|
|
vSliceIds[i] = nCopy
|
|
table.insert( vEraseIds, nCopy)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
local vLayIds = EgtGetNameInGroup( s_nPartId, SLICE_LAYER .. '*')
|
|
|
|
for i = 1, #vSliceIds do
|
|
-- effettuo lo slicing in layer separati ( analogamente a quanto avviene con il solido)
|
|
local nResId
|
|
if nSlicingType ~= SLICING_TYPE.MULTIPLANAR and nSlicingType ~= SLICING_TYPE.MULTIPLANAR_DEG45 and nSlicingType ~= SLICING_TYPE.MULTIPLANAR_HOR then
|
|
nResId = EgtParPlanesSurfTmInters( ORIG(), vSlicingDir[1], vSlicingVal, vSliceIds[i], s_nPartId, GDB_RT.GLOB, TOLER)
|
|
else
|
|
for j = 1, #vSlicingVal do
|
|
local vtN = vSlicingDir[j]
|
|
local ptOn = vSlicingVal[j]
|
|
if j == 1 then
|
|
ptOn = ptOn + DELTAZ * vtN
|
|
end
|
|
|
|
local nGrpId = EgtGroup( s_nPartId)
|
|
if not nResId then
|
|
nResId = nGrpId
|
|
end
|
|
EgtPlaneSurfTmInters( ptOn, vtN, vSliceIds[i], nGrpId, GDB_RT.GLOB, TOLER)
|
|
end
|
|
end
|
|
|
|
-- recupero i risultati e li sposto nei layers opportuni
|
|
local j = 1
|
|
while nResId do
|
|
local vtSlicing = vSlicingDir[j] or vSlicingDir[1]
|
|
|
|
-- recupero il gruppo nel layer dove salvare lo slicing degli oggetti o lo creo se non esiste
|
|
local nGrp = EgtGetFirstNameInGroup( vLayIds[j], sNameGrp)
|
|
if not nGrp then
|
|
nGrp = EgtGroup( vLayIds[j])
|
|
EgtSetName( nGrp, sNameGrp)
|
|
EgtSetStatus( nGrp, GDB_ST.OFF)
|
|
end
|
|
|
|
-- verifico se necessario ricalcolo
|
|
local nId = EgtGetLastInGroup( nResId)
|
|
local bRecalc = false
|
|
local vtRecalc
|
|
while nId and not bRecalc do
|
|
local nGeoType = EgtGetType( nId)
|
|
if nGeoType == GDB_TY.SRF_MESH then
|
|
bRecalc = true
|
|
vtRecalc = EgtSurfTmFacetNormVersor( nId, 0)
|
|
elseif nGeoType == GDB_TY.GEO_POINT then
|
|
bRecalc = true
|
|
elseif nType ~= TYPE.RIB and nType ~= TYPE.INFILL and not EgtCurveIsClosed( nId) then
|
|
bRecalc = true
|
|
end
|
|
nId = EgtGetPrev( nId)
|
|
end
|
|
|
|
-- eseguo eventuale ricalcolo
|
|
local dDeltaZ = EgtIf( j == 1, DELTAZ, 0)
|
|
if bRecalc then
|
|
EgtEmptyGroup( nResId)
|
|
dDeltaZ = dDeltaZ + EgtIf( vtRecalc and vtRecalc:getZ() > 0, - DELTAZ, DELTAZ)
|
|
-- eseguo il ricalcolo solo a quella quota
|
|
local ptSlicing = EgtGetInfo( vLayIds[j], KEY_SLICE_POS, 'p')
|
|
local ptOn = ptSlicing + dDeltaZ * vtSlicing
|
|
EgtPlaneSurfTmInters( ptOn, vtSlicing, vSliceIds[i], nResId, GDB_RT.GLOB, TOLER)
|
|
EgtOutLog( 'Warning : recalc at layer '.. EgtNumToString( j))
|
|
end
|
|
|
|
-- recupero tutte le curve
|
|
local vAllIds = EgtGetAllInGroup( nResId)
|
|
local vCrvIds = {}
|
|
for k = 1, #vAllIds do
|
|
local nGeoType = EgtGetType( nId)
|
|
if nGeoType ~= GDB_TY.SRF_MESH and nGeoType ~= GDB_TY.GEO_POINT then
|
|
table.insert( vCrvIds, vAllIds[k])
|
|
end
|
|
end
|
|
|
|
if #vCrvIds > 0 then
|
|
-- concateno le curve, le rinomino, correggo di DeltaZ e assegno i parametri
|
|
local nChainId, nCnt = EgtCurveCompoByChain( nGrp, vCrvIds, ORIG(), true, GDB_RT.LOC, TOLER)
|
|
for nId = nChainId, nChainId + nCnt - 1 do
|
|
EgtSetName( nId, sName .. EgtNumToString( i))
|
|
EgtMove( nId, - dDeltaZ * vtSlicing)
|
|
EgtSetInfo( nId, KEY_ORIGINAL_SURF, vSliceIds[i])
|
|
if tabParams then
|
|
for sKey, sVal in pairs( tabParams[i]) do
|
|
EgtSetInfo( nId, sKey, sVal)
|
|
end
|
|
end
|
|
EgtModifyCurveExtrusion( nId, vtSlicing, GDB_RT.GLOB)
|
|
EgtApproxCurve( nId, GDB_CA.ARCS, MID_TOLER)
|
|
|
|
-- se ho ancora curve aperte, segnalo errore
|
|
if nType ~= TYPE.RIB and nType ~= TYPE.INFILL and not EgtCurveIsClosed( nId) then
|
|
EgtOutLog( 'Error : hole in object (layer '.. EgtNumToString( j) ..') - CalcSlices')
|
|
table.insert( s_vErr, j)
|
|
end
|
|
end
|
|
end
|
|
|
|
local nOldId = nResId
|
|
nResId = EgtGetNext( nResId)
|
|
j = j + 1
|
|
EgtErase( nOldId)
|
|
end
|
|
end
|
|
|
|
-- rimuovo eventuali oggetti creati per lo slicing
|
|
EgtErase( vEraseIds)
|
|
|
|
return true
|
|
end
|
|
|
|
------------------------------------------------------------------------
|
|
local function SlicingInfill( nLay, sName, sNameGrp)
|
|
|
|
-- recupero tutte le curve di infill suddivise per direzioni di riferimento
|
|
local vInfillGrps = EgtGetAllInGroup( nLay)
|
|
local tIds = {}
|
|
for i = 1, #vInfillGrps do
|
|
local vIds = EgtGetAllInGroup( vInfillGrps[i])
|
|
table.insert( tIds, vIds)
|
|
end
|
|
local nGrps = #vInfillGrps
|
|
|
|
local vLayIds = EgtGetNameInGroup( s_nPartId, SLICE_LAYER .. '*')
|
|
for i = 1, #vLayIds do
|
|
|
|
-- individuo il gruppo di cui fare slicing
|
|
local nGrpIdx = ( i - 1) % nGrps + 1
|
|
|
|
-- recupero i dati di slicing
|
|
local ptSlicing = EgtGetInfo( vLayIds[i], KEY_SLICE_POS, 'p')
|
|
local vtSlicing = EgtGetInfo( vLayIds[i], KEY_SLICE_DIR, 'v')
|
|
|
|
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
|
|
|
|
local vInfillIds = tIds[nGrpIdx]
|
|
for j = 1, #vInfillIds do
|
|
-- verifico che oggetto sia trimesh
|
|
local nTypeObj = EgtGetType( vInfillIds[j])
|
|
if nTypeObj == GDB_TY.SRF_MESH then
|
|
-- slicing
|
|
local nNewId, nPntCnt, nCrvCnt, nSrfCnt = EgtPlaneSurfTmInters( ptSlicing, vtSlicing, vInfillIds[j], nGrp, GDB_RT.GLOB, TOLER)
|
|
|
|
-- rimuovo punti
|
|
for nId = nNewId, nNewId + nPntCnt -1 do
|
|
EgtErase( nId)
|
|
end
|
|
|
|
-- concateno le curve
|
|
local vChain = {}
|
|
for nId = nNewId + nPntCnt, nNewId + nPntCnt + nCrvCnt - 1 do
|
|
table.insert( vChain, nId)
|
|
end
|
|
local nChainId, nCnt = EgtCurveCompoByChain( nGrp, vChain, ORIG(), true, GDB_RT.LOC, TOLER)
|
|
-- sistemo le curve
|
|
for nId = nChainId, nChainId + nCnt - 1 do
|
|
EgtSetName( nId, sName .. EgtNumToString(j))
|
|
EgtSetInfo( nId, KEY_ORIGINAL_SURF, vInfillIds[j])
|
|
EgtModifyCurveExtrusion( nId, vtSlicing, GDB_RT.GLOB)
|
|
EgtApproxCurve( nId, GDB_CA.ARCS, MID_TOLER)
|
|
end
|
|
|
|
-- rimuovo superfici
|
|
for nId = nNewId + nPntCnt + nCrvCnt, nNewId + nPntCnt + nCrvCnt + nSrfCnt - 1 do
|
|
EgtErase( nId)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
return true
|
|
end
|
|
|
|
|
|
----------------------------------------------------------------------
|
|
----------------------------- INFILL ---------------------------------
|
|
----------------------------------------------------------------------
|
|
local function CalcLinesInfill( nInfillGrp, nStmId, dOffsStm, frLoc, vtOffs, dDist)
|
|
|
|
-- box locale
|
|
local b3Loc = EgtGetBBoxRef( nStmId, GDB_BB.STANDARD, frLoc)
|
|
b3Loc:expand( dOffsStm)
|
|
local ptMin = b3Loc:getMin()
|
|
local ptMax = b3Loc:getMax()
|
|
local dZ = ptMin:getZ()
|
|
|
|
local nGrp = EgtGroup( nInfillGrp, frLoc, GDB_RT.GLOB)
|
|
|
|
-- calcolo la prima quota a cui tracciare la linea
|
|
local dY = ptMin:getY() + vtOffs:getY() % dDist
|
|
|
|
-- calcolo le linee
|
|
while dY < ptMax:getY() + GEO.EPS_SMALL do
|
|
local ptS = Point3d( ptMin:getX(), dY, dZ)
|
|
local ptE = Point3d( ptMax:getX(), dY, dZ)
|
|
EgtCurveCompoFromPoints( nGrp, {ptS, ptE})
|
|
dY = dY + dDist
|
|
end
|
|
end
|
|
|
|
--------------------------------------------------------------------
|
|
local function CalcSimpleGridInfill( nInfillGrp, nStmId, dOffsStm, frLoc, vtOffs, dDist)
|
|
|
|
-- box locale
|
|
local b3Loc = EgtGetBBoxRef( nStmId, GDB_BB.STANDARD, frLoc)
|
|
b3Loc:expand( dOffsStm)
|
|
local ptMin = b3Loc:getMin()
|
|
local ptMax = b3Loc:getMax()
|
|
local dZ = ptMin:getZ()
|
|
|
|
-- direzione principale
|
|
local nMainGrp = EgtGroup( nInfillGrp, frLoc, GDB_RT.GLOB)
|
|
local dY = ptMin:getY() + vtOffs:getY() % dDist
|
|
while dY < ptMax:getY() + GEO.EPS_SMALL do
|
|
local ptS = Point3d( ptMin:getX(), dY, dZ)
|
|
local ptE = Point3d( ptMax:getX(), dY, dZ)
|
|
EgtCurveCompoFromPoints( nMainGrp, {ptS, ptE})
|
|
dY = dY + dDist
|
|
end
|
|
|
|
-- direzione secondaria ( ortogonale alla principale)
|
|
local nOtherGrp = EgtGroup( nInfillGrp, frLoc, GDB_RT.GLOB)
|
|
local dX = ptMin:getX() + vtOffs:getX() % dDist
|
|
while dX < ptMax:getX() + GEO.EPS_SMALL do
|
|
local ptS = Point3d( dX, ptMin:getY(), dZ)
|
|
local ptE = Point3d( dX, ptMax:getY(), dZ)
|
|
EgtCurveCompoFromPoints( nOtherGrp, {ptS, ptE})
|
|
dX = dX + dDist
|
|
end
|
|
end
|
|
|
|
--------------------------------------------------------------------
|
|
local function CompletePatternBaseCrv( nCrvBase, dAng, dDim1, dDim2, dMinY, dMaxY)
|
|
|
|
local vtDir1 = VectorFromPolar( 1, dAng)
|
|
local vtDir2 = VectorFromPolar( 1, 180 - dAng)
|
|
|
|
-- completo la curva base in Y+
|
|
local ptNext = EgtEP( nCrvBase)
|
|
while ptNext:getY() < dMaxY do
|
|
ptNext = ptNext + dDim1 * vtDir1
|
|
EgtAddCurveCompoLine( nCrvBase, ptNext)
|
|
ptNext = ptNext + dDim2 * Y_AX()
|
|
EgtAddCurveCompoLine( nCrvBase, ptNext)
|
|
ptNext = ptNext + dDim1 * vtDir2
|
|
EgtAddCurveCompoLine( nCrvBase, ptNext)
|
|
ptNext = ptNext + dDim2 * Y_AX()
|
|
EgtAddCurveCompoLine( nCrvBase, ptNext)
|
|
end
|
|
|
|
-- completo la curva base in Y-
|
|
ptNext = EgtSP( nCrvBase)
|
|
while ptNext:getY() > dMinY do
|
|
ptNext = ptNext - dDim1 * vtDir2
|
|
EgtAddCurveCompoLine( nCrvBase, ptNext, false)
|
|
ptNext = ptNext - dDim2 * Y_AX()
|
|
EgtAddCurveCompoLine( nCrvBase, ptNext, false)
|
|
ptNext = ptNext - dDim1 * vtDir1
|
|
EgtAddCurveCompoLine( nCrvBase, ptNext, false)
|
|
ptNext = ptNext - dDim2 * Y_AX()
|
|
EgtAddCurveCompoLine( nCrvBase, ptNext, false)
|
|
end
|
|
|
|
return nCrvBase
|
|
end
|
|
|
|
--------------------------------------------------------------------
|
|
local function CalcPatternCrvs( nGrp, nCrvBase, dMinX, dMaxX, dAng, dDim1, dStrand)
|
|
|
|
local ptRef = EgtSP( nCrvBase)
|
|
local dReflDist = dDim1 * cos( dAng) + dStrand * 0.5
|
|
|
|
-- rifletto la curva base in X+
|
|
local nPrevCrv = nCrvBase
|
|
local dXRefl = ptRef:getX() + dReflDist
|
|
while dXRefl < dMaxX do
|
|
local nCurrCrv = EgtCopyGlob( nPrevCrv, nGrp, GDB_IN.LAST_SON)
|
|
-- riflessione
|
|
local ptRefl = Point3d( dXRefl, ptRef:getY(), ptRef:getZ())
|
|
EgtMirror( nCurrCrv, ptRefl, X_AX())
|
|
-- aggiorno per iterazione successiva
|
|
nPrevCrv = nCurrCrv
|
|
dXRefl = dXRefl + dReflDist + 0.5 * dStrand
|
|
end
|
|
|
|
-- rifletto la curva base in X-
|
|
nPrevCrv = nCrvBase
|
|
dXRefl = ptRef:getX() - dStrand / 2
|
|
while dXRefl > dMinX do
|
|
local nCurrCrv = EgtCopyGlob( nPrevCrv, nGrp, GDB_IN.FIRST_SON)
|
|
-- riflessione
|
|
local ptRefl = Point3d( dXRefl, ptRef:getY(), ptRef:getZ())
|
|
EgtMirror( nCurrCrv, ptRefl, X_AX())
|
|
-- aggiorno per iterazione successiva
|
|
nPrevCrv = nCurrCrv
|
|
dXRefl = dXRefl - dReflDist - 0.5 * dStrand
|
|
end
|
|
end
|
|
|
|
--------------------------------------------------------------------
|
|
local function CalcGridFromPattern( nInfillGrp, nStmId, dOffsStm, frRef, vtOffs, dDim1, dDim2, dAng, vSlicesAng, dStrand, vtSlicing)
|
|
|
|
-- direzione principale
|
|
local frLoc = Frame3d( frRef)
|
|
frLoc:rotate( ORIG(), vtSlicing, vSlicesAng[1])
|
|
local nMainGrp = EgtGroup( nInfillGrp, frLoc, GDB_RT.GLOB)
|
|
|
|
-- recupero il box del pezzo nel frame della direzione principale
|
|
local b3Loc = EgtGetBBoxRef( nStmId, GDB_BB.STANDARD, frLoc)
|
|
b3Loc:expand( dOffsStm)
|
|
local ptMin = b3Loc:getMin()
|
|
local ptMax = b3Loc:getMax()
|
|
|
|
-- creo la curva base
|
|
vtOffs:locToLoc( frRef, frLoc)
|
|
local pt1 = ptMin + vtOffs
|
|
local pt2 = pt1 + dDim2 * Y_AX()
|
|
local nCrvBase = EgtCurveCompoFromPoints( nMainGrp, {pt1, pt2})
|
|
if not nCrvBase then
|
|
-- caso zigzag con dOverlapGrid = 100 ( dDim2 = 0)
|
|
local vtDir1 = VectorFromPolar( 1, dAng)
|
|
local vtDir2 = VectorFromPolar( 1, 180 - dAng)
|
|
pt2 = pt1 + vtDir1 * dDim1
|
|
nCrvBase = EgtCurveCompoFromPoints( nMainGrp, {pt1, pt2})
|
|
pt2 = pt2 + vtDir2 * dDim1
|
|
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)
|
|
|
|
-- calcolo centro di rotazione
|
|
local dDimPattern = 2 * dDim2 + 2 * dDim1 * sin( dAng)
|
|
local dDimTot = ( EgtEP( nCrvBase):getY() - EgtSP( nCrvBase):getY()) - dDim2
|
|
local dExtra = floor( dDimTot / dDimPattern * 0.5) * dDimPattern
|
|
local ptCen = EgtSP( nCrvBase) + ( 3/2 * dDim2 + sin( dAng) * dDim1) * Y_AX() - dStrand * 0.5 * X_AX() + dExtra * Y_AX()
|
|
|
|
for i = 2, #vSlicesAng do
|
|
|
|
-- calcolo il nuovo riferimento
|
|
local frLoc2 = Frame3d( frRef)
|
|
frLoc2:rotate( ORIG(), vtSlicing, vSlicesAng[i])
|
|
local nGrp = EgtGroup( nInfillGrp, frLoc2, GDB_RT.GLOB)
|
|
|
|
-- calcolo il box del pezzo nel nuovo riferimento
|
|
local b3Loc2 = EgtGetBBoxRef( nStmId, GDB_BB.STANDARD, frLoc2)
|
|
b3Loc2:expand( dOffsStm)
|
|
local ptMin2 = b3Loc2:getMin()
|
|
local ptMax2 = b3Loc2:getMax()
|
|
|
|
-- creo curva base ruotando la curva base della direzione principale
|
|
local nCrvBase2 = EgtCopyGlob( nCrvBase, nMainGrp)
|
|
EgtRotate( nCrvBase2, ptCen, Z_AX(), vSlicesAng[i] - vSlicesAng[1])
|
|
EgtRelocateGlob( nCrvBase2, nGrp)
|
|
CompletePatternBaseCrv( nCrvBase2, dAng, dDim1, dDim2, ptMin2:getY(), ptMax2:getY())
|
|
|
|
-- preparo le altre curve
|
|
CalcPatternCrvs( nGrp, nCrvBase2, ptMin2:getX(), ptMax2:getX(), dAng, dDim1, dStrand)
|
|
end
|
|
|
|
end
|
|
|
|
--------------------------------------------------------------------
|
|
local function CalcInfill( nInfillGrp, nType, dDensity, dDir, dOffsX, dOffsY, dGridOverlap, nStmId, dOffsStm, dStrand, vtSlicing)
|
|
|
|
-- frame locale alla direzione dell'infill
|
|
local frLoc = Frame3d( ORIG(), vtSlicing)
|
|
frLoc:rotate( ORIG(), vtSlicing, dDir)
|
|
|
|
-- distanza fra le varie passate
|
|
local dDist = 100 / dDensity * dStrand
|
|
|
|
-- offset in X e Y
|
|
local vtOffs = Vector3d( dOffsX, dOffsY, 0)
|
|
vtOffs:toLoc( frLoc)
|
|
|
|
-- eventuale correzione di dGridOverlap
|
|
if nType ~= FILL_TYPE.LINES and nType ~= FILL_TYPE.GRID and dDensity > 85 then
|
|
EgtOutLog( 'Warning: InfillGridOverlap is ignored due to high infill density - CalcSlices')
|
|
dGridOverlap = 0
|
|
end
|
|
|
|
-- creo le curve di infill
|
|
if nType == FILL_TYPE.LINES then
|
|
CalcLinesInfill( nInfillGrp, nStmId, dOffsStm, frLoc, vtOffs, dDist)
|
|
|
|
elseif nType == FILL_TYPE.GRID then
|
|
CalcSimpleGridInfill( nInfillGrp, nStmId, dOffsStm, frLoc, vtOffs, dDist)
|
|
|
|
elseif nType == FILL_TYPE.ZIG_ZAG_GRID then
|
|
local dRealStrand = ( 1 - dGridOverlap / 100) * dStrand
|
|
local dAng = 45
|
|
local dDim1 = ( dDist - dStrand) * sqrt(2)
|
|
local dDim2 = dRealStrand
|
|
local vSlicesAng = {-45, 45}
|
|
CalcGridFromPattern( nInfillGrp, nStmId, dOffsStm, frLoc, vtOffs, dDim1, dDim2, dAng, vSlicesAng, dRealStrand, vtSlicing)
|
|
|
|
elseif nType == FILL_TYPE.HONEYCOMB or nType == FILL_TYPE.HONEYCOMB_GRID then
|
|
local dRealStrand = ( 1 - dGridOverlap / 100) * dStrand
|
|
local dAng = 30
|
|
local dDim1 = ( dDist - dStrand) / cos(30)
|
|
local dDim2 = dDim1 + sqrt(3) * dRealStrand
|
|
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, 0.05, GDB_RT.GLOB)
|
|
end
|
|
EgtErase( vCrvs)
|
|
|
|
end
|
|
|
|
--------------------------------------------------------------------
|
|
local function PrepareInfill( nStmId, vtSlicing)
|
|
|
|
-- verifco se richiesto infill
|
|
local nShells = EgtGetInfo( s_nPartId, KEY_SHELLS_NBR, 'i')
|
|
if nShells == 0 then return end
|
|
local nType = EgtGetInfo( s_nPartId, KEY_INFILL_TYPE, 'i') or FILL_TYPE.NONE
|
|
if nType == FILL_TYPE.NONE then return end
|
|
local dDensity = EgtGetInfo( s_nPartId, KEY_INFILL_DENSITY, 'd') or 0
|
|
if dDensity < GEO.EPS_SMALL then return end
|
|
dDensity = EgtClamp( dDensity, 0, 90)
|
|
|
|
-- recupero i parametri per infill
|
|
local dDir = EgtGetInfo( s_nPartId, KEY_INFILL_DIR, 'd') or 0
|
|
local dOffsX = EgtGetInfo( s_nPartId, KEY_INFILL_OFFSET_X, 'd') or 0
|
|
local dOffsY = EgtGetInfo( s_nPartId, KEY_INFILL_OFFSET_Y, 'd') or 0
|
|
local dGridOverlap = EgtGetInfo( s_nPartId, KEY_INFILL_GRID_OVERLAP, 'd') or 0
|
|
local dOffsStm = EgtGetInfo( s_nPartId, KEY_OFFSET_SLICE, 'd')
|
|
local dStrand = EgtGetInfo( s_nPartId, KEY_INFILL_STRAND, 'd') or s_dStrand
|
|
if dStrand < GEO.EPS_SMALL then
|
|
dStrand = s_dStrand
|
|
end
|
|
|
|
-- creo gruppo per infill
|
|
local nInfillGrp = EgtGroup( s_nPartId)
|
|
EgtSetName( nInfillGrp, INFILL_GRP)
|
|
EgtSetStatus( nInfillGrp, GDB_ST.OFF)
|
|
|
|
-- calcolo infill
|
|
CalcInfill( nInfillGrp, nType, dDensity, dDir, dOffsX, dOffsY, dGridOverlap, nStmId, dOffsStm, dStrand, vtSlicing)
|
|
|
|
-- aggiungo allo slicing
|
|
SlicingInfill( nInfillGrp, INFILL_CRV, INFILL_GRP)
|
|
end
|
|
|
|
----------------------------------------------------------------------
|
|
local function PrepareAuxSolidsInfill( nSolidsLay, vtSlicing)
|
|
|
|
local vIds = EgtGetAllInGroup( nSolidsLay)
|
|
-- scorro tutti gli AuxSolids
|
|
for i = 1, #vIds do
|
|
if EgtGetType( vIds[i]) == GDB_TY.SRF_MESH then
|
|
-- verifico se da realizzare con infill e non con solidfill
|
|
local nType = ReadParam( vIds[i], KEY_AUX_SOLIDS_INFILL, 'i', FILL_TYPE.NONE)
|
|
if nType & FILL_CATEGORY.INFILL ~= 0 then
|
|
local dDensity = ReadParam( vIds[i], KEY_AUX_SOLIDS_DENSITY, 'd', 0)
|
|
if dDensity > GEO.EPS_SMALL then
|
|
dDensity = EgtClamp( dDensity, 0, 90)
|
|
|
|
-- recupero i parametri per infill
|
|
local dDir = ReadParam( vIds[i], KEY_AUX_SOLIDS_DIR, 'd', 0)
|
|
local dOffsX = ReadParam( vIds[i], KEY_AUX_SOLIDS_OFFSET_X, 'd', 0)
|
|
local dOffsY = ReadParam( vIds[i], KEY_AUX_SOLIDS_OFFSET_Y, 'd', 0)
|
|
local dGridOverlap = ReadParam( vIds[i], KEY_AUX_SOLIDS_GRID_OVERLAP, 'd', 0)
|
|
local dStrand = ReadParam( vIds[i], KEY_AUX_SOLIDS_STRAND, 'd', s_dStrand)
|
|
if dStrand < GEO.EPS_SMALL then
|
|
dStrand = s_dStrand
|
|
end
|
|
|
|
-- creo gruppo associato
|
|
local nInfillGrp = EgtGroup( s_nPartId)
|
|
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 .. EgtNumToString( vIds[i]) .. '_'
|
|
SlicingInfill( nInfillGrp, sName, AUX_SOLIDS_GRP)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
--------------------------------------------------------------------
|
|
local function ExtractRibsLoops( nRibsGrp, nStmId)
|
|
|
|
local nLoopGrp = EgtGroup( s_nPartId)
|
|
EgtSetName( nLoopGrp, RIBS_LOOP_GRP)
|
|
EgtSetStatus( nLoopGrp, GDB_ST.OFF)
|
|
|
|
-- recupero tutti i setti
|
|
local vIds = EgtGetAllInGroup( nRibsGrp)
|
|
for i = 1, #vIds do
|
|
-- se trimesh estraggo i contorni
|
|
if EgtGetType( vIds[i]) == GDB_TY.SRF_MESH then
|
|
local nCrv, nCnt = EgtExtractSurfTmLoops( vIds[i], nLoopGrp)
|
|
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 .. EgtNumToString( vIds[i]))
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
---------------------------------------------------------------------
|
|
local function ValueInArray( vArr, nValue)
|
|
for _, val in ipairs( vArr) do
|
|
if val == nValue then
|
|
return true
|
|
end
|
|
end
|
|
return false
|
|
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 = V_NULL()
|
|
local ptSlicePrev = ORIG()
|
|
local bFirst = true
|
|
local dTotMaxDist = dSliceStep
|
|
local dTotMinDist = dSliceStep
|
|
local nLayMin = 1
|
|
local nLayMax = 1
|
|
for i = 1, #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)
|
|
|
|
if #vCrvs > 0 and not bFirst then
|
|
-- verifico la distanza dal piano precedente lungo vtSlice
|
|
local dMinDist = GEO.INFINITO
|
|
local dMaxDist = - GEO.INFINITO
|
|
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
|
|
|
|
if dDist < dMinDist then
|
|
dMinDist = dDist
|
|
end
|
|
if dDist > dMaxDist then
|
|
dMaxDist = dDist
|
|
end
|
|
end
|
|
end
|
|
EgtSetInfo( vLayers[i], KEY_MULTIPLANAR_MAX_H, dMaxDist)
|
|
EgtSetInfo( vLayers[i], KEY_MULTIPLANAR_MIN_H, dMinDist)
|
|
|
|
if dMinDist < dTotMinDist then
|
|
dTotMinDist = dMinDist
|
|
nLayMin = i
|
|
end
|
|
if dMaxDist > dTotMaxDist then
|
|
dTotMaxDist = dMaxDist
|
|
nLayMax = i
|
|
end
|
|
end
|
|
|
|
-- aggiorno per layer successivo
|
|
vtSlicePrev = vtSlice
|
|
ptSlicePrev = ptSlice
|
|
if #vCrvs > 0 then
|
|
bFirst = false
|
|
end
|
|
end
|
|
|
|
EgtOutLog( 'MinStrandH = ' .. EgtNumToString( dTotMinDist) .. ' (layer ' .. EgtNumToString( nLayMin) .. ')')
|
|
EgtOutLog( 'MaxStrandH = ' .. EgtNumToString( dTotMaxDist) .. ' (layer ' .. EgtNumToString( nLayMax) .. ')')
|
|
|
|
return true
|
|
end
|
|
|
|
---------------------------------------------------------------------
|
|
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, #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 or nSlicingType == SLICING_TYPE.MULTIPLANAR_DEG45 or nSlicingType == SLICING_TYPE.MULTIPLANAR_HOR then
|
|
vtSlicing = vSlicingDir[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)
|
|
|
|
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + i / #vSlicingVal * 100, 0) == 1 then return false end
|
|
end
|
|
|
|
-- slicing dei setti
|
|
SlicingExtraObjects( nRibsLay, vSlicingDir, vSlicingVal, nSlicingType, TYPE.RIB, RIBS_CRV, RIBS_GRP)
|
|
|
|
-- se slicing multplanare verifico altezza massima e strand
|
|
if nSlicingType == SLICING_TYPE.MULTIPLANAR or nSlicingType == SLICING_TYPE.MULTIPLANAR_DEG45 or nSlicingType == SLICING_TYPE.MULTIPLANAR_HOR then
|
|
if dMaxH < GEO.INFINITO then
|
|
local vLayIds = EgtGetNameInGroup( s_nPartId, SLICE_LAYER .. '*')
|
|
for i = 1, #vLayIds do
|
|
local nCurrRibsLay = EgtGetFirstNameInGroup( vLayIds[i], RIBS_GRP)
|
|
local b3Box = EgtGetBBoxGlob( nCurrRibsLay, GDB_BB.STANDARD)
|
|
local ptMax = b3Box:getMax() + dSliceStep * vSlicingDir[i]
|
|
if ptMax:getZ() > dMaxH + GEO.EPS_SMALL then
|
|
-- elimino tutti i layer da questo punto
|
|
for j = i, #vLayIds do
|
|
EgtErase( vLayIds[j])
|
|
end
|
|
break
|
|
end
|
|
end
|
|
end
|
|
VerifyMultiPlanarStrand()
|
|
end
|
|
|
|
return true
|
|
end
|
|
|
|
---------------------------------------------------------------------
|
|
local function MultiPlanarSlicing( nStmId, dMaxH, vSlicingDir, vSlicingPnt)
|
|
|
|
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 SlicingWithSolid( nStmId, nSlicingType, dMaxH, vSlicingVal, vSlicingDir)
|
|
|
|
-- calcolo delle intersezioni
|
|
local nLayId, nCnt
|
|
if nSlicingType ~= SLICING_TYPE.MULTIPLANAR and nSlicingType ~= SLICING_TYPE.MULTIPLANAR_DEG45 and nSlicingType ~= SLICING_TYPE.MULTIPLANAR_HOR 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 = nCnt + 12
|
|
|
|
local nFirstSolidLay -- primo layer che contiene il solido
|
|
local nLastSolidLay -- ultimo layer che contiene il solido
|
|
|
|
-- scorro i risultati dello slicing
|
|
while nLayId do
|
|
|
|
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 or nSlicingType == SLICING_TYPE.MULTIPLANAR_DEG45 or nSlicingType == SLICING_TYPE.MULTIPLANAR_HOR 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
|
|
local vtRecalc
|
|
local nRecalc = 0
|
|
while nId and not bRecalc do
|
|
local nType = EgtGetType( nId)
|
|
if nType == GDB_TY.SRF_MESH then
|
|
bRecalc = true
|
|
vtRecalc = EgtSurfTmFacetNormVersor( nId, 0)
|
|
elseif nType == GDB_TY.GEO_POINT then
|
|
bRecalc = true
|
|
end
|
|
nId = EgtGetPrev( nId)
|
|
end
|
|
-- eseguo eventuale ricalcolo
|
|
::RECALC::
|
|
if bRecalc then
|
|
nRecalc = nRecalc + 1
|
|
EgtEmptyGroup( nLayId)
|
|
dDeltaZ = dDeltaZ + EgtIf( vtRecalc and vtRecalc:getZ() > 0, - DELTAZ, DELTAZ)
|
|
-- eseguo il ricalcolo solo a quella quota
|
|
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)
|
|
|
|
-- salvo i risultati nel layer
|
|
local vClosedId = {}
|
|
local vOpenId = {}
|
|
nId = EgtGetFirstInGroup( nLayId)
|
|
|
|
while nId do
|
|
if EgtGetType( nId) == GDB_TY.GEO_POINT or EgtGetType( nId) == GDB_TY.SRF_MESH then
|
|
-- se punto o superficie lo elimino
|
|
EgtErase( nId)
|
|
else
|
|
-- correggo eventuali spostamenti in Z
|
|
if abs( dDeltaZ) > GEO.EPS_SMALL then
|
|
EgtMove( nId, - dDeltaZ * vtSlicing)
|
|
end
|
|
-- verifico presenza contorni aperti
|
|
if EgtCurveIsClosed( nId) then
|
|
table.insert( vClosedId, nId)
|
|
else
|
|
table.insert( vOpenId, nId)
|
|
end
|
|
end
|
|
|
|
nId = EgtGetNext( nId)
|
|
end
|
|
|
|
-- se presenti percorsi aperti, provo a concatenarli
|
|
if #vOpenId > 0 then
|
|
local nRordId, nRordCnt = EgtCurveCompoByReorder( nLayId, vOpenId, ORIG(), true, GDB_RT.LOC, BIG_TOLER)
|
|
if nRordId then
|
|
vOpenId = {}
|
|
for nId = nRordId, nRordId + nRordCnt - 1 do
|
|
if EgtCurveIsClosed( nId) then
|
|
table.insert( vClosedId, nId)
|
|
else
|
|
table.insert( vOpenId, nId)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
-- chiudo percorsi aperti praticamente chiusi ed elimino quelli troppo corti
|
|
for i = #vOpenId, 1, -1 do
|
|
local ptStart = EgtSP( vOpenId[i])
|
|
local ptEnd = EgtEP( vOpenId[i])
|
|
if EgtCurveLength( vOpenId[i]) < MIN_LEN then
|
|
EgtErase( vOpenId[i])
|
|
table.remove( vOpenId, i)
|
|
elseif AreSamePointEpsilon( ptStart, ptEnd, BIG_TOLER) then
|
|
local ptMid = ( ptStart + ptEnd) / 2
|
|
EgtModifyCurveStartPoint( vOpenId[i], ptMid)
|
|
EgtModifyCurveEndPoint( vOpenId[i], ptMid)
|
|
table.insert( vClosedId, vOpenId[i])
|
|
table.remove( vOpenId, i)
|
|
end
|
|
end
|
|
local bAllClosed = ( #vOpenId == 0)
|
|
-- elimino percorsi chiusi di area troppo piccola
|
|
for i = #vClosedId, 1, -1 do
|
|
local _, _, dArea = EgtCurveArea( vClosedId[i])
|
|
if dArea and dArea < MIN_AREA then
|
|
EgtErase( vClosedId[i])
|
|
table.remove( vClosedId, i)
|
|
end
|
|
end
|
|
|
|
-- se necessario e possibile, torno a ricalcolo
|
|
if not bAllClosed and nRecalc < 2 then
|
|
bRecalc = true
|
|
goto RECALC
|
|
end
|
|
|
|
-- imposto dati ausiliari
|
|
EgtSetName( nLayId, EgtIf( bAllClosed, '', '__') .. SLICE_LAYER .. EgtNumToString( nLayCnt))
|
|
EgtSetInfo( nLayId, 'CrvCnt', #vClosedId + #vOpenId)
|
|
EgtSetColor( vClosedId or {}, 'TEAL')
|
|
EgtSetColor( vOpenId or {}, 'RED')
|
|
|
|
-- creo flat region a partire dalle curve chiuse ottenute con lo slicing
|
|
for i = 1, #vClosedId do
|
|
EgtModifyCurveExtrusion( vClosedId[i], vtSlicing, GDB_RT.GLOB)
|
|
EgtApproxCurve( vClosedId[i], GDB_CA.ARCS, MID_TOLER)
|
|
end
|
|
local nSurfFR, nSrfNbr = EgtSurfFlatRegion( nLayId, vClosedId)
|
|
|
|
if nSurfFR then
|
|
|
|
-- se slicing del solido valido aggiorno i valori dei layer estremi che lo contengono
|
|
if not nFirstSolidLay then nFirstSolidLay = nLayCnt end
|
|
nLastSolidLay = nLayCnt
|
|
|
|
-- verifico orientamento della normale
|
|
local vtN = EgtSurfFrNormVersor( nSurfFR, GDB_ID.ROOT)
|
|
if AreOppositeVectorApprox( vtN, vtSlicing) then
|
|
EgtInvertSurf( nSurfFR)
|
|
end
|
|
-- elimino le curve originali
|
|
for i = 1, #vClosedId do
|
|
EgtErase( vClosedId[i])
|
|
end
|
|
-- esplodo in superfici elementari
|
|
local vSrfId = {}
|
|
for j = 1, nSrfNbr do
|
|
local nCurrSfrId = nSurfFR + j - 1
|
|
local nSfrId, nSfrCnt = EgtExplodeSurf( nCurrSfrId)
|
|
for nId = nSfrId, nSfrId + nSfrCnt - 1 do
|
|
table.insert( vSrfId, nId)
|
|
end
|
|
end
|
|
-- calcolo i centri delle superfici
|
|
local vCen = {}
|
|
for j = 1, #vSrfId do
|
|
local nId = vSrfId[j]
|
|
local ptCen = EgtCP( nId, GDB_ID.ROOT)
|
|
table.insert( vCen, ptCen)
|
|
end
|
|
-- ordino le superfici rispetto ai centri del layer precedente
|
|
local vOrd = {}
|
|
if #vPrevCen > 1 then
|
|
for j = 1, #vPrevCen do
|
|
local nMin = 0
|
|
local dMinDist = GEO.INFINITO
|
|
for k = 1, #vCen do
|
|
if not ValueInArray( vOrd, k) then
|
|
local dDist = dist( vPrevCen[j], vCen[k])
|
|
if dDist < dMinDist then
|
|
dMinDist = dDist
|
|
nMin = k
|
|
end
|
|
end
|
|
end
|
|
if nMin ~= 0 then
|
|
vOrd[j] = nMin
|
|
end
|
|
end
|
|
-- se necessario, completo il vettore di ordine
|
|
if #vOrd < #vSrfId then
|
|
for j = 1, #vSrfId do
|
|
if not ValueInArray( vOrd, j) then
|
|
table.insert( vOrd, j)
|
|
end
|
|
end
|
|
end
|
|
else
|
|
EgtSpInit()
|
|
for j = 1, #vSrfId do
|
|
EgtSpAddPoint( vCen[j]:getX(), vCen[j]:getY())
|
|
end
|
|
vOrd = EgtSpCalculate( SHP_TY.OPEN)
|
|
EgtSpTerminate()
|
|
if not vOrd then
|
|
for j = 1, #vSrfId do
|
|
vOrd[j] = j
|
|
end
|
|
end
|
|
end
|
|
-- salvo i nuovi centri
|
|
for j = 1, #vSrfId do
|
|
vPrevCen[j] = vCen[vOrd[j]]
|
|
end
|
|
-- creo i gruppi di percorsi
|
|
for j = 1, #vSrfId do
|
|
local nId = vSrfId[vOrd[j]]
|
|
-- per ogni chunk creo un gruppo di percorsi
|
|
local nGrpCrv = EgtGroup( nLayId)
|
|
EgtSetName( nGrpCrv, CONTOUR_GRP .. EgtNumToString( j, 1))
|
|
EgtRelocateGlob( nId, nGrpCrv)
|
|
EgtSetName( nId, LAYER_SRF)
|
|
EgtSetStatus( nId, GDB_ST.OFF)
|
|
-- estraggo i contorni della superficie
|
|
local nCsId, nCsCnt = EgtExtractSurfFrChunkLoops( nId, 0, nGrpCrv)
|
|
for nId2 = nCsId, nCsId + nCsCnt - 1 do
|
|
EgtSetName( nId2, OUTER_CRV)
|
|
EgtSetStatus( nId2, GDB_ST.OFF)
|
|
EgtSetColor( nId2, 'BLACK')
|
|
end
|
|
end
|
|
|
|
if #vOpenId > 1 then
|
|
EgtOutLog( 'Error : hole in solid (layer '.. EgtNumToString( nLayCnt) ..') - CalcSlices')
|
|
table.insert( s_vErr, nLayCnt)
|
|
end
|
|
|
|
else
|
|
|
|
local bErr = true
|
|
-- se nessuna curva
|
|
if #vClosedId == 0 and #vOpenId == 0 then
|
|
bErr = false
|
|
-- se una sola curva, ne verifico larghezza media
|
|
elseif #vClosedId == 1 then
|
|
local dLen = EgtCurveLength( vClosedId[1])
|
|
local _, _, dArea = EgtCurveArea( vClosedId[1])
|
|
if not dArea or dArea / dLen < MID_TOLER then bErr = false end
|
|
end
|
|
|
|
-- se vero errore lo segnalo
|
|
if bErr then
|
|
EgtOutLog( 'Error : hole in solid (layer '.. EgtNumToString( nLayCnt) ..') - CalcSlices')
|
|
table.insert( s_vErr, nLayCnt)
|
|
-- cambio nome al layer
|
|
EgtSetName( nLayId, '__' .. SLICE_LAYER .. EgtNumToString( nLayCnt))
|
|
else
|
|
-- altrimenti creo il CrvGrp da utilizzare nel CalcPath
|
|
local nGrpCrv = EgtGroup( nLayId)
|
|
EgtSetName( nGrpCrv, CONTOUR_GRP .. EgtNumToString( 1))
|
|
end
|
|
end
|
|
|
|
-- passo al layer successivo
|
|
if EgtProcessEvents( EgtIf( PRINT, 100, 0) + nLayCnt / nCounterTot * 100, 0) == 1 then
|
|
-- elimino i layer non ancora analizzati
|
|
local nNext = EgtGetNext( nLayId)
|
|
while nNext do
|
|
local nCurr = nNext
|
|
nNext = EgtGetNext( nCurr)
|
|
EgtErase( nCurr)
|
|
end
|
|
return false
|
|
end
|
|
|
|
nLayCnt = nLayCnt + 1
|
|
nLayId = EgtGetNext( nLayId)
|
|
end
|
|
|
|
-- costolature
|
|
local nRibsLay = EgtGetFirstNameInGroup( s_nPartId, LAY_RIBS)
|
|
ExtractRibsLoops( nRibsLay, nStmId)
|
|
SlicingExtraObjects( nRibsLay, vSlicingDir, vSlicingVal, nSlicingType, 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
|
|
if nSlicingType ~= SLICING_TYPE.MULTIPLANAR and nSlicingType ~= SLICING_TYPE.MULTIPLANAR_DEG45 and nSlicingType ~= SLICING_TYPE.MULTIPLANAR_HOR 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, vSlicingDir, vSlicingVal, nSlicingType, 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, vSlicingDir, vSlicingVal, nSlicingType, TYPE.AUX_SOLID, AUX_SOLIDS_CRV, AUX_SOLIDS_GRP)
|
|
if nSlicingType ~= SLICING_TYPE.MULTIPLANAR and nSlicingType ~= SLICING_TYPE.MULTIPLANAR_DEG45 and nSlicingType ~= SLICING_TYPE.MULTIPLANAR_HOR 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 or nSlicingType == SLICING_TYPE.MULTIPLANAR_DEG45 or nSlicingType == SLICING_TYPE.MULTIPLANAR_HOR then
|
|
if not VerifyMultiPlanarStrand() then
|
|
return false
|
|
end
|
|
end
|
|
|
|
return true
|
|
end
|
|
|
|
---------------------------------------------------------------------
|
|
function CalcSlices.Exec( nPartId, nStmId)
|
|
|
|
s_nPartId = nPartId
|
|
|
|
local sMachIni = EgtGetCurrMachineDir() .. '\\' .. EgtGetCurrMachineName() .. '.ini'
|
|
|
|
-- recupero la direzione dello slicing
|
|
local vtSlicing = Z_AX()
|
|
local nSlicingType = EgtGetInfo( s_nPartId, KEY_SLICING_TYPE, 'i')
|
|
if nSlicingType == SLICING_TYPE.DEG45_X then
|
|
local nDir = EgtGetNumberFromIni( '3dPrinting', 'Dir45degX', 0, sMachIni)
|
|
local dHorAng = EgtIf( nDir == -1, 180, 0)
|
|
vtSlicing = VectorFromSpherical( 1, 45, dHorAng)
|
|
elseif nSlicingType == SLICING_TYPE.DEG45_Y then
|
|
local nDir = EgtGetNumberFromIni( '3dPrinting', 'Dir45degY', 0, sMachIni)
|
|
local dHorAng = EgtIf( nDir == 2, 90, -90)
|
|
vtSlicing = VectorFromSpherical( 1, 45, dHorAng)
|
|
elseif nSlicingType == SLICING_TYPE.HORIZONTAL then
|
|
vtSlicing = X_AX()
|
|
elseif nSlicingType == SLICING_TYPE.MULTIPLANAR or nSlicingType == SLICING_TYPE.MULTIPLANAR_DEG45 or nSlicingType == SLICING_TYPE.MULTIPLANAR_HOR then
|
|
-- la direzione può cambiare da un piano all'altro
|
|
vtSlicing = Z_AX()
|
|
end
|
|
|
|
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 = {}
|
|
if nStmId then table.insert( vRefIds, nStmId) end
|
|
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)
|
|
if nType == RIB_TYPE.EXTERNAL or ( nType == RIB_TYPE.UNBOUNDED and not bLimitUnbddRibs) then
|
|
table.insert( vRefIds, vRibsIds[i])
|
|
end
|
|
end
|
|
end
|
|
if #vRefIds == 0 then
|
|
EgtOutBox( "No part to be sliced!", 'Error', 'ERROR')
|
|
return false
|
|
end
|
|
|
|
-- calcolo i dati dello slicing
|
|
local dMaxH = ComputeMaxH( vRefIds, vtSlicing, dSliceStep)
|
|
local vSlicingVal = {}
|
|
local vSlicingDir = {}
|
|
if nSlicingType == SLICING_TYPE.MULTIPLANAR or nSlicingType == SLICING_TYPE.MULTIPLANAR_DEG45 or nSlicingType == SLICING_TYPE.MULTIPLANAR_HOR then
|
|
vSlicingVal, vSlicingDir = ComputeMultiplanarSlicingData( dSliceStep, nMaxSlicesNbr, nSlicingType)
|
|
else
|
|
vSlicingVal, vSlicingDir = ComputeSlicingData( vRefIds, vtSlicing, dSliceStep, dMaxH, nMaxSlicesNbr)
|
|
end
|
|
if not vSlicingVal then
|
|
return false
|
|
end
|
|
|
|
-- Eseguo slicing
|
|
local bOk = true
|
|
if not nStmId then
|
|
-- caso senza solido e solo setti
|
|
bOk = SlicingNoSolid( nRibsLay, nSlicingType, dMaxH, dSliceStep, vSlicingVal, vSlicingDir)
|
|
else
|
|
-- caso con solido
|
|
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
|
|
|
|
return bOk
|
|
end
|
|
|
|
---------------------------------------------------------------------
|
|
return CalcSlices
|