* Python3 compatibility
- Compatible with FreeCAD 0.18 with Python3 - Added 'About' under EM menu - New icon for the FH Solver - Preparing for VoxHenry - Minor fixes in the comments
This commit is contained in:
parent
85041325c3
commit
081ec9fc65
49
EM.py
49
EM.py
|
@ -38,12 +38,22 @@ __url__ = "http://www.fastfieldsolvers.com"
|
||||||
|
|
||||||
'''The E.M. module provides tools for ElectroMagnetic analysis'''
|
'''The E.M. module provides tools for ElectroMagnetic analysis'''
|
||||||
|
|
||||||
|
import sys
|
||||||
|
|
||||||
|
# Python3 compatibility
|
||||||
|
if sys.version_info >= (3, 4):
|
||||||
|
from importlib import reload
|
||||||
|
elif sys.version_info >= (3, 0):
|
||||||
|
from imp import reload
|
||||||
|
|
||||||
import FreeCAD
|
import FreeCAD
|
||||||
if FreeCAD.GuiUp:
|
if FreeCAD.GuiUp:
|
||||||
import FreeCADGui
|
import FreeCADGui
|
||||||
FreeCADGui.updateLocale()
|
FreeCADGui.updateLocale()
|
||||||
|
|
||||||
from EM_Globals import *
|
from EM_Globals import *
|
||||||
|
from EM_About import *
|
||||||
|
# FastHenry specific
|
||||||
from EM_FHNode import *
|
from EM_FHNode import *
|
||||||
from EM_FHSegment import *
|
from EM_FHSegment import *
|
||||||
from EM_FHPath import *
|
from EM_FHPath import *
|
||||||
|
@ -53,4 +63,43 @@ from EM_FHPort import *
|
||||||
from EM_FHEquiv import *
|
from EM_FHEquiv import *
|
||||||
from EM_FHSolver import *
|
from EM_FHSolver import *
|
||||||
from EM_FHInputFile import *
|
from EM_FHInputFile import *
|
||||||
|
## VoxHenry specific
|
||||||
|
#from EM_VHSolver import *
|
||||||
|
|
||||||
|
# for debugging
|
||||||
|
#import EM_Globals
|
||||||
|
#reload(EM_Globals)
|
||||||
|
#from EM_Globals import *
|
||||||
|
#import EM_About
|
||||||
|
#reload(EM_About)
|
||||||
|
#from EM_About import *
|
||||||
|
#import EM_FHNode
|
||||||
|
#reload(EM_FHNode)
|
||||||
|
#from EM_FHNode import *
|
||||||
|
#import EM_FHSegment
|
||||||
|
#reload(EM_FHSegment)
|
||||||
|
#from EM_FHSegment import *
|
||||||
|
#import EM_FHPath
|
||||||
|
#reload(EM_FHPath)
|
||||||
|
#from EM_FHPath import *
|
||||||
|
#import EM_FHPlaneHole
|
||||||
|
#reload(EM_FHPlaneHole)
|
||||||
|
#from EM_FHPlaneHole import *
|
||||||
|
#import EM_FHPlane
|
||||||
|
#reload(EM_FHPlane)
|
||||||
|
#from EM_FHPlane import *
|
||||||
|
#import EM_FHPort
|
||||||
|
#reload(EM_FHPort)
|
||||||
|
#from EM_FHPort import *
|
||||||
|
#import EM_FHEquiv
|
||||||
|
#reload(EM_FHEquiv)
|
||||||
|
#from EM_FHEquiv import *
|
||||||
|
#import EM_FHSolver
|
||||||
|
#reload(EM_FHSolver)
|
||||||
|
#from EM_FHSolver import *
|
||||||
|
#import EM_FHInputFile
|
||||||
|
#reload(EM_FHInputFile)
|
||||||
|
#from EM_FHInputFile import *
|
||||||
|
#import EM_VHSolver
|
||||||
|
#reload(EM_VHSolver)
|
||||||
|
#from EM_VHSolver import *
|
||||||
|
|
82
EM_About.py
Normal file
82
EM_About.py
Normal file
|
@ -0,0 +1,82 @@
|
||||||
|
#***************************************************************************
|
||||||
|
#* *
|
||||||
|
#* Copyright (c) 2018 * *
|
||||||
|
#* FastFieldSolvers S.R.L. *
|
||||||
|
#* http://www.fastfieldsolvers.com *
|
||||||
|
#* *
|
||||||
|
#* This program is free software; you can redistribute it and/or modify *
|
||||||
|
#* it under the terms of the GNU Lesser General Public License (LGPL) *
|
||||||
|
#* as published by the Free Software Foundation; either version 2 of *
|
||||||
|
#* the License, or (at your option) any later version. *
|
||||||
|
#* for detail see the LICENCE text file. *
|
||||||
|
#* *
|
||||||
|
#* This program is distributed in the hope that it will be useful, *
|
||||||
|
#* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
||||||
|
#* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
||||||
|
#* GNU Library General Public License for more details. *
|
||||||
|
#* *
|
||||||
|
#* You should have received a copy of the GNU Library General Public *
|
||||||
|
#* License along with this program; if not, write to the Free Software *
|
||||||
|
#* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 *
|
||||||
|
#* USA *
|
||||||
|
#* *
|
||||||
|
#***************************************************************************
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
__title__="FreeCAD E.M. Workbench About class"
|
||||||
|
__author__ = "FastFieldSolvers S.R.L."
|
||||||
|
__url__ = "http://www.fastfieldsolvers.com"
|
||||||
|
|
||||||
|
# imported defines
|
||||||
|
from EM_Globals import EM_VERSION
|
||||||
|
|
||||||
|
# defines
|
||||||
|
# about information
|
||||||
|
EM_AUTHOR = 'Copyright 2019 FastFieldSolvers S.R.L. and Efficient Power Conversion Inc.\nhttp://www.fastfieldsolvers.com, http://epc-co.com\nPartially developed by FastFieldSolvers S.R.L. under contract by EPC Inc.\n'
|
||||||
|
EM_LICENSE = 'Licensed under GNU Lesser General Public License (LGPL) version 2\n'
|
||||||
|
|
||||||
|
import FreeCAD, FreeCADGui, Mesh, Part, MeshPart, Draft, DraftGeomUtils, os
|
||||||
|
from FreeCAD import Vector
|
||||||
|
|
||||||
|
if FreeCAD.GuiUp:
|
||||||
|
import FreeCADGui
|
||||||
|
from PySide import QtCore, QtGui
|
||||||
|
from DraftTools import translate
|
||||||
|
from PySide.QtCore import QT_TRANSLATE_NOOP
|
||||||
|
else:
|
||||||
|
# \cond
|
||||||
|
def translate(ctxt,txt, utf8_decode=False):
|
||||||
|
return txt
|
||||||
|
def QT_TRANSLATE_NOOP(ctxt,txt):
|
||||||
|
return txt
|
||||||
|
# \endcond
|
||||||
|
|
||||||
|
__dir__ = os.path.dirname(__file__)
|
||||||
|
iconPath = os.path.join( __dir__, 'Resources' )
|
||||||
|
|
||||||
|
|
||||||
|
class _CommandAbout:
|
||||||
|
''' The EM About command definition
|
||||||
|
'''
|
||||||
|
def GetResources(self):
|
||||||
|
# no need of icon or accelerator
|
||||||
|
return {'MenuText': QT_TRANSLATE_NOOP("EM_About","About"),
|
||||||
|
'ToolTip': QT_TRANSLATE_NOOP("EM_About","About the ElectroMagnetic Workbench")}
|
||||||
|
|
||||||
|
def IsActive(self):
|
||||||
|
return not FreeCAD.ActiveDocument is None
|
||||||
|
|
||||||
|
def Activated(self):
|
||||||
|
msg = translate("EM","ElectroMagnetic Workbench version ") + EM_VERSION + "\n\n" + EM_AUTHOR + "\n" + EM_LICENSE
|
||||||
|
|
||||||
|
if FreeCAD.GuiUp:
|
||||||
|
# Simple QMessageBox providing "about" informaiton
|
||||||
|
diag = QtGui.QMessageBox(QtGui.QMessageBox.Information, translate("EM_About","About ElectroMagnetic workbench"), msg)
|
||||||
|
diag.setWindowModality(QtCore.Qt.ApplicationModal)
|
||||||
|
diag.exec_()
|
||||||
|
else:
|
||||||
|
FreeCAD.Console.PrintWarning(msg)
|
||||||
|
|
||||||
|
if FreeCAD.GuiUp:
|
||||||
|
FreeCADGui.addCommand('EM_About',_CommandAbout())
|
|
@ -101,12 +101,12 @@ class _FHEquiv:
|
||||||
'''
|
'''
|
||||||
if obj.Node1 == None:
|
if obj.Node1 == None:
|
||||||
return
|
return
|
||||||
elif Draft.getType(obj.Node1) <> "FHNode":
|
elif Draft.getType(obj.Node1) != "FHNode":
|
||||||
FreeCAD.Console.PrintWarning(translate("EM","Node1 is not a FHNode"))
|
FreeCAD.Console.PrintWarning(translate("EM","Node1 is not a FHNode"))
|
||||||
return
|
return
|
||||||
if obj.Node2 == None:
|
if obj.Node2 == None:
|
||||||
return
|
return
|
||||||
elif Draft.getType(obj.Node2) <> "FHNode":
|
elif Draft.getType(obj.Node2) != "FHNode":
|
||||||
FreeCAD.Console.PrintWarning(translate("EM","Node2 is not a FHNode"))
|
FreeCAD.Console.PrintWarning(translate("EM","Node2 is not a FHNode"))
|
||||||
return
|
return
|
||||||
# and finally, if everything is ok, make and assign the shape
|
# and finally, if everything is ok, make and assign the shape
|
||||||
|
|
|
@ -161,7 +161,7 @@ class _FHPath:
|
||||||
# of the vertexes, and the segments cross-section orientation will be
|
# of the vertexes, and the segments cross-section orientation will be
|
||||||
# calculated in absolute coordinates from the Positions rotations.
|
# calculated in absolute coordinates from the Positions rotations.
|
||||||
# This last part is different from FHSegment.
|
# This last part is different from FHSegment.
|
||||||
if obj.Placement <> FreeCAD.Placement():
|
if obj.Placement != FreeCAD.Placement():
|
||||||
obj.Placement = FreeCAD.Placement()
|
obj.Placement = FreeCAD.Placement()
|
||||||
# define nodes and segments
|
# define nodes and segments
|
||||||
edges_raw = []
|
edges_raw = []
|
||||||
|
|
|
@ -164,8 +164,8 @@ class _FHPlane:
|
||||||
obj.addProperty("App::PropertyLength","segwid2","EM",QT_TRANSLATE_NOOP("App::Property","Width of segments along the plane width direction ('segwid2' plane parameter)"))
|
obj.addProperty("App::PropertyLength","segwid2","EM",QT_TRANSLATE_NOOP("App::Property","Width of segments along the plane width direction ('segwid2' plane parameter)"))
|
||||||
obj.addProperty("App::PropertyLinkList","Nodes","EM",QT_TRANSLATE_NOOP("App::Property","Nodes for connections to the plane"))
|
obj.addProperty("App::PropertyLinkList","Nodes","EM",QT_TRANSLATE_NOOP("App::Property","Nodes for connections to the plane"))
|
||||||
obj.addProperty("App::PropertyLinkList","Holes","EM",QT_TRANSLATE_NOOP("App::Property","Holes in the plane"))
|
obj.addProperty("App::PropertyLinkList","Holes","EM",QT_TRANSLATE_NOOP("App::Property","Holes in the plane"))
|
||||||
obj.addProperty("App::PropertyBool","FineMesh","Component",QT_TRANSLATE_NOOP("App::Property","Specifies if this the plane fine mesh is shown (i.e. composing segments)"))
|
obj.addProperty("App::PropertyBool","FineMesh","EM",QT_TRANSLATE_NOOP("App::Property","Specifies if this the plane fine mesh is shown (i.e. composing segments)"))
|
||||||
obj.addProperty("App::PropertyBool","ShowNodes","Component",QT_TRANSLATE_NOOP("App::Property","Show the internal node grid supporting the plane"))
|
obj.addProperty("App::PropertyBool","ShowNodes","EM",QT_TRANSLATE_NOOP("App::Property","Show the internal node grid supporting the plane"))
|
||||||
obj.Proxy = self
|
obj.Proxy = self
|
||||||
self.Type = "FHPlane"
|
self.Type = "FHPlane"
|
||||||
self.FineMesh = False
|
self.FineMesh = False
|
||||||
|
@ -232,7 +232,7 @@ class _FHPlane:
|
||||||
obj.segwid2 = segwid2
|
obj.segwid2 = segwid2
|
||||||
FreeCAD.Console.PrintWarning(translate("EM","Plane segwid2 would cause segments overlap, re-setting segwid2 to the maximum possible"))
|
FreeCAD.Console.PrintWarning(translate("EM","Plane segwid2 would cause segments overlap, re-setting segwid2 to the maximum possible"))
|
||||||
# if needed, apply the same Placement of the Base object to the FHPlane object
|
# if needed, apply the same Placement of the Base object to the FHPlane object
|
||||||
if obj.Placement <> obj.Base.Placement:
|
if obj.Placement != obj.Base.Placement:
|
||||||
obj.Placement = obj.Base.Placement
|
obj.Placement = obj.Base.Placement
|
||||||
# Start node and hole repositioning in relative coordinate system of the conductive plane
|
# Start node and hole repositioning in relative coordinate system of the conductive plane
|
||||||
#
|
#
|
||||||
|
@ -241,14 +241,14 @@ class _FHPlane:
|
||||||
# that once a node is child of a plane, it cannot be moved independently by changing
|
# that once a node is child of a plane, it cannot be moved independently by changing
|
||||||
# its placement
|
# its placement
|
||||||
for node in obj.Nodes:
|
for node in obj.Nodes:
|
||||||
if node.Placement <> obj.Placement:
|
if node.Placement != obj.Placement:
|
||||||
node.Placement = obj.Placement
|
node.Placement = obj.Placement
|
||||||
# These holes have already been adopted by the plane, if they are in the obj.Holes list;
|
# These holes have already been adopted by the plane, if they are in the obj.Holes list;
|
||||||
# therefore, must just make sure they track the plane placement. Also, this assures
|
# therefore, must just make sure they track the plane placement. Also, this assures
|
||||||
# that once a hole is child of a plane, it cannot be moved independently by changing
|
# that once a hole is child of a plane, it cannot be moved independently by changing
|
||||||
# its placement
|
# its placement
|
||||||
for hole in obj.Holes:
|
for hole in obj.Holes:
|
||||||
if hole.Placement <> obj.Placement:
|
if hole.Placement != obj.Placement:
|
||||||
hole.Placement = obj.Placement
|
hole.Placement = obj.Placement
|
||||||
# Check if the user selected a coarse or a fine mesh.
|
# Check if the user selected a coarse or a fine mesh.
|
||||||
if obj.FineMesh == False:
|
if obj.FineMesh == False:
|
||||||
|
|
|
@ -99,12 +99,12 @@ class _FHPort:
|
||||||
'''
|
'''
|
||||||
if obj.NodePos == None:
|
if obj.NodePos == None:
|
||||||
return
|
return
|
||||||
elif Draft.getType(obj.NodePos) <> "FHNode":
|
elif Draft.getType(obj.NodePos) != "FHNode":
|
||||||
FreeCAD.Console.PrintWarning(translate("EM","NodePos is not a FHNode"))
|
FreeCAD.Console.PrintWarning(translate("EM","NodePos is not a FHNode"))
|
||||||
return
|
return
|
||||||
if obj.NodeNeg == None:
|
if obj.NodeNeg == None:
|
||||||
return
|
return
|
||||||
elif Draft.getType(obj.NodeNeg) <> "FHNode":
|
elif Draft.getType(obj.NodeNeg) != "FHNode":
|
||||||
FreeCAD.Console.PrintWarning(translate("EM","NodeNeg is not a FHNode"))
|
FreeCAD.Console.PrintWarning(translate("EM","NodeNeg is not a FHNode"))
|
||||||
return
|
return
|
||||||
if obj.NodePos == obj.NodeNeg:
|
if obj.NodePos == obj.NodeNeg:
|
||||||
|
@ -246,7 +246,7 @@ class _CommandFHPort:
|
||||||
endNode = selobj.Object
|
endNode = selobj.Object
|
||||||
else:
|
else:
|
||||||
FreeCAD.Console.PrintWarning(translate("EM","More than two FHNodes selected when creating a FHPort. Using only the first two."))
|
FreeCAD.Console.PrintWarning(translate("EM","More than two FHNodes selected when creating a FHPort. Using only the first two."))
|
||||||
if startNode <> None and endNode <> None:
|
if startNode != None and endNode != None:
|
||||||
FreeCAD.ActiveDocument.openTransaction(translate("EM","Create FHPort"))
|
FreeCAD.ActiveDocument.openTransaction(translate("EM","Create FHPort"))
|
||||||
FreeCADGui.addModule("EM")
|
FreeCADGui.addModule("EM")
|
||||||
FreeCADGui.doCommand('obj=EM.makeFHPort(nodePos=FreeCAD.ActiveDocument.'+startNode.Name+',nodeNeg=FreeCAD.ActiveDocument.'+endNode.Name+')')
|
FreeCADGui.doCommand('obj=EM.makeFHPort(nodePos=FreeCAD.ActiveDocument.'+startNode.Name+',nodeNeg=FreeCAD.ActiveDocument.'+endNode.Name+')')
|
||||||
|
|
|
@ -137,12 +137,12 @@ class _FHSegment:
|
||||||
#FreeCAD.Console.PrintWarning("_FHSegment execute()\n") #debug
|
#FreeCAD.Console.PrintWarning("_FHSegment execute()\n") #debug
|
||||||
if obj.NodeStart == None:
|
if obj.NodeStart == None:
|
||||||
return
|
return
|
||||||
elif Draft.getType(obj.NodeStart) <> "FHNode":
|
elif Draft.getType(obj.NodeStart) != "FHNode":
|
||||||
FreeCAD.Console.PrintWarning(translate("EM","NodeStart is not a FHNode"))
|
FreeCAD.Console.PrintWarning(translate("EM","NodeStart is not a FHNode"))
|
||||||
return
|
return
|
||||||
if obj.NodeEnd == None:
|
if obj.NodeEnd == None:
|
||||||
return
|
return
|
||||||
elif Draft.getType(obj.NodeEnd) <> "FHNode":
|
elif Draft.getType(obj.NodeEnd) != "FHNode":
|
||||||
FreeCAD.Console.PrintWarning(translate("EM","NodeEnd is not a FHNode"))
|
FreeCAD.Console.PrintWarning(translate("EM","NodeEnd is not a FHNode"))
|
||||||
return
|
return
|
||||||
# the FHSegment has no Placement in itself:
|
# the FHSegment has no Placement in itself:
|
||||||
|
@ -154,7 +154,7 @@ class _FHSegment:
|
||||||
# So let's keep the FHSegment placement at zero, and use the FHNodes to move the segment position.
|
# So let's keep the FHSegment placement at zero, and use the FHNodes to move the segment position.
|
||||||
# This is also necessary as we should not change the segment cross-section orientation
|
# This is also necessary as we should not change the segment cross-section orientation
|
||||||
# using the Placement, otherwise it will not be relative to the global axis system
|
# using the Placement, otherwise it will not be relative to the global axis system
|
||||||
if obj.Placement <> FreeCAD.Placement():
|
if obj.Placement != FreeCAD.Placement():
|
||||||
obj.Placement = FreeCAD.Placement()
|
obj.Placement = FreeCAD.Placement()
|
||||||
# check if we have a 'Base' object; if so, if segment end nodes
|
# check if we have a 'Base' object; if so, if segment end nodes
|
||||||
# were already defined, re-set them according to the 'Base' object;
|
# were already defined, re-set them according to the 'Base' object;
|
||||||
|
@ -170,13 +170,13 @@ class _FHSegment:
|
||||||
return
|
return
|
||||||
# ok, it's valid. Let's verify if this is a Wire.
|
# ok, it's valid. Let's verify if this is a Wire.
|
||||||
if Draft.getType(obj.Base) == "Wire":
|
if Draft.getType(obj.Base) == "Wire":
|
||||||
if obj.NodeStart <> None:
|
if obj.NodeStart != None:
|
||||||
abs_pos = obj.NodeStart.Proxy.getAbsCoord()
|
abs_pos = obj.NodeStart.Proxy.getAbsCoord()
|
||||||
# 'obj.Base.Start' is an absolute position
|
# 'obj.Base.Start' is an absolute position
|
||||||
# if 'NodeStart' is not in that position, move it
|
# if 'NodeStart' is not in that position, move it
|
||||||
if (abs_pos-obj.Base.Start).Length > EMFHSEGMENT_LENTOL:
|
if (abs_pos-obj.Base.Start).Length > EMFHSEGMENT_LENTOL:
|
||||||
obj.NodeStart.Proxy.setAbsCoord(obj.Base.Start)
|
obj.NodeStart.Proxy.setAbsCoord(obj.Base.Start)
|
||||||
if obj.NodeEnd <> None:
|
if obj.NodeEnd != None:
|
||||||
abs_pos = obj.NodeEnd.Proxy.getAbsCoord()
|
abs_pos = obj.NodeEnd.Proxy.getAbsCoord()
|
||||||
# 'obj.Base.Start' is an absolute position
|
# 'obj.Base.Start' is an absolute position
|
||||||
# if 'NodeStart' is not in that position, move it
|
# if 'NodeStart' is not in that position, move it
|
||||||
|
@ -328,7 +328,7 @@ class _CommandFHSegment:
|
||||||
endNode = selobj.Object
|
endNode = selobj.Object
|
||||||
else:
|
else:
|
||||||
FreeCAD.Console.PrintWarning(translate("EM","More than two FHNodes selected when creating a FHSegment. Using only the first two."))
|
FreeCAD.Console.PrintWarning(translate("EM","More than two FHNodes selected when creating a FHSegment. Using only the first two."))
|
||||||
if startNode <> None and endNode <> None:
|
if startNode != None and endNode != None:
|
||||||
FreeCAD.ActiveDocument.openTransaction(translate("EM","Create FHSegment"))
|
FreeCAD.ActiveDocument.openTransaction(translate("EM","Create FHSegment"))
|
||||||
FreeCADGui.addModule("EM")
|
FreeCADGui.addModule("EM")
|
||||||
FreeCADGui.doCommand('obj=EM.makeFHSegment(nodeStart=FreeCAD.ActiveDocument.'+startNode.Name+',nodeEnd=FreeCAD.ActiveDocument.'+endNode.Name+')')
|
FreeCADGui.doCommand('obj=EM.makeFHSegment(nodeStart=FreeCAD.ActiveDocument.'+startNode.Name+',nodeEnd=FreeCAD.ActiveDocument.'+endNode.Name+')')
|
||||||
|
|
|
@ -87,8 +87,8 @@ def makeFHSolver(units=None,sigma=None,nhinc=None,nwinc=None,rh=None,rw=None,fmi
|
||||||
'ndec' is the float value defining how many frequency points per decade
|
'ndec' is the float value defining how many frequency points per decade
|
||||||
will be simulated
|
will be simulated
|
||||||
'folder' is the folder into which the FastHenry file will be saved.
|
'folder' is the folder into which the FastHenry file will be saved.
|
||||||
Defaults to the user's home path (e.g. in Windows "C:\Documents
|
Defaults to the user's home path (e.g. in Windows "C:\\Documents
|
||||||
and Settings\username\My Documents", in Linux "/home/username")
|
and Settings\\username\\My Documents", in Linux "/home/username")
|
||||||
'filename' is the name of the file that will be exported.
|
'filename' is the name of the file that will be exported.
|
||||||
Defaults to EMFHSOLVER_DEF_FILENAME
|
Defaults to EMFHSOLVER_DEF_FILENAME
|
||||||
'name' is the name of the object
|
'name' is the name of the object
|
||||||
|
@ -98,7 +98,7 @@ def makeFHSolver(units=None,sigma=None,nhinc=None,nwinc=None,rh=None,rw=None,fmi
|
||||||
obj = FreeCAD.ActiveDocument.addObject("Part::FeaturePython", name)
|
obj = FreeCAD.ActiveDocument.addObject("Part::FeaturePython", name)
|
||||||
obj.Label = translate("EM", name)
|
obj.Label = translate("EM", name)
|
||||||
# this adds the relevant properties to the object
|
# this adds the relevant properties to the object
|
||||||
#'obj' (e.g. 'Base' property) making it a _FHSegment
|
#'obj' (e.g. 'Base' property) making it a _FHSolver
|
||||||
_FHSolver(obj)
|
_FHSolver(obj)
|
||||||
# manage ViewProvider object
|
# manage ViewProvider object
|
||||||
if FreeCAD.GuiUp:
|
if FreeCAD.GuiUp:
|
||||||
|
|
|
@ -33,6 +33,8 @@ from FreeCAD import Vector
|
||||||
|
|
||||||
# defines
|
# defines
|
||||||
#
|
#
|
||||||
|
# version information
|
||||||
|
EM_VERSION = '1.0.1'
|
||||||
# default node color
|
# default node color
|
||||||
EMFHNODE_DEF_NODECOLOR = (1.0,0.0,0.0)
|
EMFHNODE_DEF_NODECOLOR = (1.0,0.0,0.0)
|
||||||
# tolerance in degrees when verifying if vectors are parallel
|
# tolerance in degrees when verifying if vectors are parallel
|
||||||
|
|
|
@ -42,8 +42,11 @@ class EMWorkbench(Workbench):
|
||||||
# import the EM module (and therefore all commands makeXXX)
|
# import the EM module (and therefore all commands makeXXX)
|
||||||
import EM
|
import EM
|
||||||
# E.M. tools
|
# E.M. tools
|
||||||
|
self.emtools = ["EM_About"]
|
||||||
self.emfhtools = ["EM_FHSolver", "EM_FHNode", "EM_FHSegment", "EM_FHPath", "EM_FHPlane",
|
self.emfhtools = ["EM_FHSolver", "EM_FHNode", "EM_FHSegment", "EM_FHPath", "EM_FHPlane",
|
||||||
"EM_FHPlaneHole", "EM_FHPlaneAddRemoveNodeHole", "EM_FHEquiv", "EM_FHPort", "EM_FHInputFile"]
|
"EM_FHPlaneHole", "EM_FHPlaneAddRemoveNodeHole", "EM_FHEquiv", "EM_FHPort", "EM_FHInputFile"]
|
||||||
|
#self.emvhtools = ["EM_VHSolver"]
|
||||||
|
self.emvhtools = []
|
||||||
# draft tools
|
# draft tools
|
||||||
# setup menus
|
# setup menus
|
||||||
self.draftcmdList = ["Draft_Line","Draft_Rectangle"]
|
self.draftcmdList = ["Draft_Line","Draft_Rectangle"]
|
||||||
|
@ -59,10 +62,11 @@ class EMWorkbench(Workbench):
|
||||||
'Draft_Snap_Dimensions','Draft_Snap_WorkingPlane']
|
'Draft_Snap_Dimensions','Draft_Snap_WorkingPlane']
|
||||||
|
|
||||||
def QT_TRANSLATE_NOOP(scope, text): return text
|
def QT_TRANSLATE_NOOP(scope, text): return text
|
||||||
self.appendToolbar(QT_TRANSLATE_NOOP("Workbench","E.M. tools"),self.emfhtools)
|
self.appendToolbar(QT_TRANSLATE_NOOP("Workbench","E.M. FastHenry tools"),self.emfhtools)
|
||||||
|
#self.appendToolbar(QT_TRANSLATE_NOOP("Workbench","E.M. VoxHenry tools"),self.emvhtools)
|
||||||
self.appendToolbar(QT_TRANSLATE_NOOP("Workbench","Draft creation tools"),self.draftcmdList)
|
self.appendToolbar(QT_TRANSLATE_NOOP("Workbench","Draft creation tools"),self.draftcmdList)
|
||||||
self.appendToolbar(QT_TRANSLATE_NOOP("Workbench","Draft mod tools"),self.draftmodtools)
|
self.appendToolbar(QT_TRANSLATE_NOOP("Workbench","Draft mod tools"),self.draftmodtools)
|
||||||
self.appendMenu(QT_TRANSLATE_NOOP("EM","&EM"),self.emfhtools)
|
self.appendMenu(QT_TRANSLATE_NOOP("EM","&EM"),self.emfhtools + self.emvhtools + self.emtools)
|
||||||
self.appendMenu(QT_TRANSLATE_NOOP("EM","&Draft"),self.draftcmdList+self.draftmodtools+self.treecmdList)
|
self.appendMenu(QT_TRANSLATE_NOOP("EM","&Draft"),self.draftcmdList+self.draftmodtools+self.treecmdList)
|
||||||
self.appendMenu([QT_TRANSLATE_NOOP("EM","&Draft"),QT_TRANSLATE_NOOP("arch","Snapping")],self.snapList)
|
self.appendMenu([QT_TRANSLATE_NOOP("EM","&Draft"),QT_TRANSLATE_NOOP("arch","Snapping")],self.snapList)
|
||||||
#FreeCADGui.addIconPath(":/icons")
|
#FreeCADGui.addIconPath(":/icons")
|
||||||
|
|
19
README.md
19
README.md
|
@ -4,7 +4,7 @@
|
||||||
|
|
||||||
### FastHenry support
|
### FastHenry support
|
||||||
|
|
||||||
Copyright (c) 2018
|
Copyright (c) 2019
|
||||||
Efficient Power Conversion Corporation, Inc. http://epc-co.com
|
Efficient Power Conversion Corporation, Inc. http://epc-co.com
|
||||||
|
|
||||||
Developed by FastFieldSolvers S.R.L. http://www.fastfieldsolvers.com under contract by EPC
|
Developed by FastFieldSolvers S.R.L. http://www.fastfieldsolvers.com under contract by EPC
|
||||||
|
@ -12,7 +12,7 @@ Developed by FastFieldSolvers S.R.L. http://www.fastfieldsolvers.com under cont
|
||||||
|
|
||||||
### FasterCap and FastCap support
|
### FasterCap and FastCap support
|
||||||
|
|
||||||
Copyright (c) 2018
|
Copyright (c) 2019
|
||||||
FastFieldSolvers S.R.L. http://www.fastfieldsolvers.com
|
FastFieldSolvers S.R.L. http://www.fastfieldsolvers.com
|
||||||
|
|
||||||
## Description
|
## Description
|
||||||
|
@ -25,14 +25,25 @@ At present, the workbench supports:
|
||||||
- [FastHenry](https://www.fastfieldsolvers.com/fasthenry2.htm) inductance solver: ongoing development including GUI support
|
- [FastHenry](https://www.fastfieldsolvers.com/fasthenry2.htm) inductance solver: ongoing development including GUI support
|
||||||
- [FasterCap](https://www.fastfieldsolvers.com/fastercap.htm) capacitance solver: ongoing development, today at the stage of Python macro only, for creating an input file
|
- [FasterCap](https://www.fastfieldsolvers.com/fastercap.htm) capacitance solver: ongoing development, today at the stage of Python macro only, for creating an input file
|
||||||
|
|
||||||
|
## Version
|
||||||
|
|
||||||
|
The current version of the ElectroMagnetic workbench can be shown, once installed, from the **EM** menu, selecting **About**.
|
||||||
|
|
||||||
|
The version number is also reported in the sources, in the global variable **EM_VERSION** in the file EM_Globals.py.
|
||||||
|
|
||||||
## Installing
|
## Installing
|
||||||
|
|
||||||
The ElectroMagnetic workbench is managed as a FreeCAD addon. This addon can be downloaded by clicking the **Download ZIP** button found on top of the page, or using **Git**. The addon must be placed in your user's FreeCAD/Mod folder.
|
The ElectroMagnetic workbench is managed as a FreeCAD addon. It can be installed from within FreeCAD using the add-ons manager under the <b>Tools</b> menu, see the [FreeCAD add-on documentation](https://www.freecadweb.org/wiki/Std_AddonMgr) for more specific instructions
|
||||||
|
|
||||||
|
### Manual installation
|
||||||
|
|
||||||
|
This addon can also be manually installed, but this method is not recommended, as the add-ons manager provides a more user friendly experience.
|
||||||
|
|
||||||
|
If you still wish to manually install the workbench, you can download it by clicking the **Download ZIP** button found on top of the page, or using **Git**. The addon must be un-zipped in your user's FreeCAD/Mod folder.
|
||||||
|
|
||||||
**Note**: Your user's FreeCAD folder location is obtained by typing in FreeCAD's python console: `FreeCAD.ConfigGet("UserAppData")`.
|
**Note**: Your user's FreeCAD folder location is obtained by typing in FreeCAD's python console: `FreeCAD.ConfigGet("UserAppData")`.
|
||||||
|
|
||||||
You must then rename the new folder 'EM'. I.e., the new folder structure must be <UserAppData>/Mod/EM. Opening, or closing/reopening FreeCAD then reloads the workbenches, and the E.M. workbench will show up in the pull-down workbenches menu in FreeCAD.
|
You must then rename the new folder 'EM', i.e. the new folder structure must be <UserAppData>/Mod/EM. Opening, or closing/reopening FreeCAD then reloads the workbenches, and the E.M. workbench will show up in the pull-down workbenches menu in FreeCAD.
|
||||||
|
|
||||||
## Additional information
|
## Additional information
|
||||||
|
|
||||||
|
|
|
@ -91,6 +91,45 @@
|
||||||
y1="7.794034"
|
y1="7.794034"
|
||||||
x2="47.247158"
|
x2="47.247158"
|
||||||
y2="52.521305"
|
y2="52.521305"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.1846386,0,0,1.1846386,-9.501508,-5.4148589)" />
|
||||||
|
<linearGradient
|
||||||
|
gradientTransform="translate(9.9161935,2.2059659)"
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient4150"
|
||||||
|
id="linearGradient4148-7"
|
||||||
|
x1="16.556818"
|
||||||
|
y1="7.794034"
|
||||||
|
x2="47.247158"
|
||||||
|
y2="52.521305"
|
||||||
|
gradientUnits="userSpaceOnUse" />
|
||||||
|
<linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient4150"
|
||||||
|
id="linearGradient4159"
|
||||||
|
x1="16.556818"
|
||||||
|
y1="7.794034"
|
||||||
|
x2="47.247158"
|
||||||
|
y2="52.521305"
|
||||||
|
gradientUnits="userSpaceOnUse" />
|
||||||
|
<linearGradient
|
||||||
|
gradientTransform="matrix(0.50862782,0,0,0.50862782,15.408648,27.042116)"
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient4150"
|
||||||
|
id="linearGradient4148-5"
|
||||||
|
x1="16.556818"
|
||||||
|
y1="7.794034"
|
||||||
|
x2="47.247158"
|
||||||
|
y2="52.521305"
|
||||||
|
gradientUnits="userSpaceOnUse" />
|
||||||
|
<linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient4150"
|
||||||
|
id="linearGradient4172"
|
||||||
|
x1="16.556818"
|
||||||
|
y1="7.794034"
|
||||||
|
x2="47.247158"
|
||||||
|
y2="52.521305"
|
||||||
gradientUnits="userSpaceOnUse" />
|
gradientUnits="userSpaceOnUse" />
|
||||||
</defs>
|
</defs>
|
||||||
<sodipodi:namedview
|
<sodipodi:namedview
|
||||||
|
@ -101,17 +140,31 @@
|
||||||
inkscape:pageopacity="0.0"
|
inkscape:pageopacity="0.0"
|
||||||
inkscape:pageshadow="2"
|
inkscape:pageshadow="2"
|
||||||
inkscape:zoom="5.5"
|
inkscape:zoom="5.5"
|
||||||
inkscape:cx="-82.818182"
|
inkscape:cx="-22.636364"
|
||||||
inkscape:cy="26.181817"
|
inkscape:cy="26.215803"
|
||||||
inkscape:current-layer="layer1"
|
inkscape:current-layer="layer1"
|
||||||
showgrid="true"
|
showgrid="true"
|
||||||
inkscape:document-units="px"
|
inkscape:document-units="px"
|
||||||
inkscape:grid-bbox="true"
|
inkscape:grid-bbox="true"
|
||||||
inkscape:window-width="725"
|
inkscape:window-width="1920"
|
||||||
inkscape:window-height="403"
|
inkscape:window-height="1018"
|
||||||
inkscape:window-x="1097"
|
inkscape:window-x="-8"
|
||||||
inkscape:window-y="363"
|
inkscape:window-y="-8"
|
||||||
inkscape:window-maximized="0" />
|
inkscape:window-maximized="1"
|
||||||
|
inkscape:snap-grids="true"
|
||||||
|
showguides="false"
|
||||||
|
inkscape:snap-nodes="true"
|
||||||
|
inkscape:snap-others="true"
|
||||||
|
inkscape:snap-to-guides="false"
|
||||||
|
inkscape:snap-global="true">
|
||||||
|
<inkscape:grid
|
||||||
|
type="xygrid"
|
||||||
|
id="grid4187"
|
||||||
|
spacingx="1"
|
||||||
|
spacingy="1"
|
||||||
|
visible="true"
|
||||||
|
empspacing="2" />
|
||||||
|
</sodipodi:namedview>
|
||||||
<metadata
|
<metadata
|
||||||
id="metadata2865">
|
id="metadata2865">
|
||||||
<rdf:RDF>
|
<rdf:RDF>
|
||||||
|
@ -147,15 +200,48 @@
|
||||||
inkscape:groupmode="layer">
|
inkscape:groupmode="layer">
|
||||||
<text
|
<text
|
||||||
xml:space="preserve"
|
xml:space="preserve"
|
||||||
style="font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-size:64px;line-height:125%;font-family:'DejaVu Serif';-inkscape-font-specification:'DejaVu Serif';letter-spacing:0px;word-spacing:0px;fill:url(#linearGradient4148);fill-opacity:1;stroke:#241c1c;stroke-width:2;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:4.0999999"
|
style="font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-size:75.81687164px;line-height:125%;font-family:Georgia;-inkscape-font-specification:Georgia;letter-spacing:0px;word-spacing:0px;fill:url(#linearGradient4148);fill-opacity:1;stroke:#241c1c;stroke-width:2.918185;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:4.0999999"
|
||||||
x="9.272727"
|
x="1.483322"
|
||||||
y="54"
|
y="58.55563"
|
||||||
id="text3014"
|
id="text3014"
|
||||||
sodipodi:linespacing="125%"><tspan
|
sodipodi:linespacing="125%"
|
||||||
|
transform="scale(1.0619928,0.94162596)"><tspan
|
||||||
sodipodi:role="line"
|
sodipodi:role="line"
|
||||||
id="tspan3016"
|
id="tspan3016"
|
||||||
x="9.272727"
|
x="1.483322"
|
||||||
y="54"
|
y="58.55563"
|
||||||
style="fill-opacity:1;fill:url(#linearGradient4148)">S</tspan></text>
|
style="font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-family:Georgia;-inkscape-font-specification:Georgia;fill:url(#linearGradient4148);fill-opacity:1;stroke-width:2.918185;stroke-miterlimit:4;stroke-dasharray:none">S</tspan></text>
|
||||||
|
<text
|
||||||
|
xml:space="preserve"
|
||||||
|
style="font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-size:64px;line-height:125%;font-family:'DejaVu Serif';-inkscape-font-specification:'DejaVu Serif';letter-spacing:0px;word-spacing:0px;fill:url(#linearGradient4148-7);fill-opacity:1;stroke:#241c1c;stroke-width:2;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:4.0999999"
|
||||||
|
x="19.188921"
|
||||||
|
y="56.205967"
|
||||||
|
id="text3014-3"
|
||||||
|
sodipodi:linespacing="125%"><tspan
|
||||||
|
sodipodi:role="line"
|
||||||
|
id="tspan3016-8"
|
||||||
|
x="19.188921"
|
||||||
|
y="56.205967"
|
||||||
|
style="fill:url(#linearGradient4148-7);fill-opacity:1" /></text>
|
||||||
|
<rect
|
||||||
|
style="fill:#ffffff;fill-opacity:0.85789472;stroke:none;stroke-width:2;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1"
|
||||||
|
id="rect4204"
|
||||||
|
width="45.81818"
|
||||||
|
height="32"
|
||||||
|
x="16.181818"
|
||||||
|
y="30" />
|
||||||
|
<text
|
||||||
|
xml:space="preserve"
|
||||||
|
style="font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-size:32.55217743px;line-height:125%;font-family:Georgia;-inkscape-font-specification:Georgia;letter-spacing:0px;word-spacing:0px;fill:url(#linearGradient4148-5);fill-opacity:1;stroke:#241c1c;stroke-width:2;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:4.0999999"
|
||||||
|
x="20.125015"
|
||||||
|
y="54.508018"
|
||||||
|
id="text3014-6"
|
||||||
|
sodipodi:linespacing="125%"
|
||||||
|
transform="scale(0.91885746,1.0883081)"><tspan
|
||||||
|
sodipodi:role="line"
|
||||||
|
id="tspan3016-1"
|
||||||
|
x="20.125015"
|
||||||
|
y="54.508018"
|
||||||
|
style="font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-family:Georgia;-inkscape-font-specification:Georgia;fill:url(#linearGradient4148-5);fill-opacity:1;stroke-width:2;stroke-miterlimit:4;stroke-dasharray:none">FH</tspan></text>
|
||||||
</g>
|
</g>
|
||||||
</svg>
|
</svg>
|
||||||
|
|
Before Width: | Height: | Size: 5.0 KiB After Width: | Height: | Size: 8.7 KiB |
|
@ -27,6 +27,7 @@
|
||||||
|
|
||||||
import FreeCAD, Mesh, Part, MeshPart, DraftGeomUtils, os
|
import FreeCAD, Mesh, Part, MeshPart, DraftGeomUtils, os
|
||||||
from FreeCAD import Vector
|
from FreeCAD import Vector
|
||||||
|
import numpy as np
|
||||||
|
|
||||||
if FreeCAD.GuiUp:
|
if FreeCAD.GuiUp:
|
||||||
import FreeCADGui
|
import FreeCADGui
|
||||||
|
@ -562,25 +563,25 @@ def meshSolidWithSegments(obj=None,delta=1.0,deltaX=0.0,deltaY=0.0,deltaZ=0.0,st
|
||||||
for step_y in range(0,stepsY+1):
|
for step_y in range(0,stepsY+1):
|
||||||
for step_x in range(0,stepsX):
|
for step_x in range(0,stepsX):
|
||||||
# if the node and the next are inside the object shape, create the segment
|
# if the node and the next are inside the object shape, create the segment
|
||||||
if nodes[step_x,step_y,step_z] <> None and nodes[step_x+1,step_y,step_z] <> None:
|
if nodes[step_x,step_y,step_z] != None and nodes[step_x+1,step_y,step_z] != None:
|
||||||
segment = EM_FHSegment.makeFHSegment(nodeStart=nodes[step_x,step_y,step_z],nodeEnd=nodes[step_x+1,step_y,step_z],width=deltaX,height=deltaZ)
|
segment = EM_FHSegment.makeFHSegment(nodeStart=nodes[step_x,step_y,step_z],nodeEnd=nodes[step_x+1,step_y,step_z],width=deltaX,height=deltaZ)
|
||||||
# then along y
|
# then along y
|
||||||
for step_z in range(0,stepsZ+1):
|
for step_z in range(0,stepsZ+1):
|
||||||
for step_x in range(0,stepsX+1):
|
for step_x in range(0,stepsX+1):
|
||||||
for step_y in range(0,stepsY):
|
for step_y in range(0,stepsY):
|
||||||
# if the node and the next are inside the object shape, create the segment
|
# if the node and the next are inside the object shape, create the segment
|
||||||
if nodes[step_x,step_y,step_z] <> None and nodes[step_x,step_y+1,step_z] <> None:
|
if nodes[step_x,step_y,step_z] != None and nodes[step_x,step_y+1,step_z] != None:
|
||||||
segment = EM_FHSegment.makeFHSegment(nodeStart=nodes[step_x,step_y,step_z],nodeEnd=nodes[step_x,step_y+1,step_z],width=deltaY,height=deltaZ)
|
segment = EM_FHSegment.makeFHSegment(nodeStart=nodes[step_x,step_y,step_z],nodeEnd=nodes[step_x,step_y+1,step_z],width=deltaY,height=deltaZ)
|
||||||
# finally along z
|
# finally along z
|
||||||
for step_x in range(0,stepsX+1):
|
for step_x in range(0,stepsX+1):
|
||||||
for step_y in range(0,stepsY+1):
|
for step_y in range(0,stepsY+1):
|
||||||
for step_z in range(0,stepsZ):
|
for step_z in range(0,stepsZ):
|
||||||
# if the node and the next are inside the object shape, create the segment
|
# if the node and the next are inside the object shape, create the segment
|
||||||
if nodes[step_x,step_y,step_z] <> None and nodes[step_x,step_y,step_z+1] <> None:
|
if nodes[step_x,step_y,step_z] != None and nodes[step_x,step_y,step_z+1] != None:
|
||||||
segment = EM_FHSegment.makeFHSegment(nodeStart=nodes[step_x,step_y,step_z],nodeEnd=nodes[step_x,step_y,step_z+1],width=deltaX,height=deltaY)
|
segment = EM_FHSegment.makeFHSegment(nodeStart=nodes[step_x,step_y,step_z],nodeEnd=nodes[step_x,step_y,step_z+1],width=deltaX,height=deltaY)
|
||||||
|
|
||||||
def meshSolidWithVoxels(obj=None,delta=1.0,stayInside=False):
|
def meshSolidWithVoxels(obj=None,delta=1.0):
|
||||||
''' Voxel a solid object
|
''' Voxelize a solid object
|
||||||
'''
|
'''
|
||||||
if obj == None:
|
if obj == None:
|
||||||
return
|
return
|
||||||
|
@ -612,73 +613,288 @@ def meshSolidWithVoxels(obj=None,delta=1.0,stayInside=False):
|
||||||
pos_y = pos_y + delta
|
pos_y = pos_y + delta
|
||||||
pos_x = pos_x + delta
|
pos_x = pos_x + delta
|
||||||
return isNode
|
return isNode
|
||||||
# if we must don't need to stay within the object shape boundaries,
|
|
||||||
# the voxel will overlap the shape contour
|
|
||||||
# nodes=np.full((stepsX+1,stepsY+1,stepsZ+1), None, np.object)
|
|
||||||
# if stayInside == False:
|
|
||||||
# pos_x = bbox.XMin + deltaSideX
|
|
||||||
# for step_x in range(0,stepsX+1):
|
|
||||||
# pos_y = bbox.YMin + deltaSideY
|
|
||||||
# for step_y in range(0,stepsY+1):
|
|
||||||
# pos_z = bbox.ZMin + deltaSideZ
|
|
||||||
# for step_z in range(0,stepsZ+1):
|
|
||||||
# # if the point is inside the object shape, or on the surface, flag it
|
|
||||||
# if isNode[step_x,step_y,step_z] == True:
|
|
||||||
# # create the node
|
|
||||||
# node = EM_FHNode.makeFHNode(X=pos_x, Y=pos_y, Z=pos_z)
|
|
||||||
# # store it in the array
|
|
||||||
# nodes[step_x,step_y,step_z] = node
|
|
||||||
# pos_z = pos_z + deltaZ
|
|
||||||
# pos_y = pos_y + deltaY
|
|
||||||
# pos_x = pos_x + deltaX
|
|
||||||
# # if we must stay within the object shape boundaries (within the accuracy
|
|
||||||
# # of the point sampling)
|
|
||||||
# else:
|
|
||||||
# pos_x = bbox.XMin + deltaSideX
|
|
||||||
# for step_x in range(0,stepsX):
|
|
||||||
# pos_y = bbox.YMin + deltaSideY
|
|
||||||
# for step_y in range(0,stepsY):
|
|
||||||
# pos_z = bbox.ZMin + deltaSideZ
|
|
||||||
# for step_z in range(0,stepsZ):
|
|
||||||
# # if all the eight cube corners are inside the object shape,
|
|
||||||
# # we consider the center point well inside the object shape, i.e. also
|
|
||||||
# # for a segment lying on a plane parallel to the plane xy,
|
|
||||||
# # with width=deltaX, height=deltaY we are within the object
|
|
||||||
# if (isNode[step_x,step_y,step_z] == True and isNode[step_x+1,step_y,step_z] == True and
|
|
||||||
# isNode[step_x,step_y+1,step_z] == True and isNode[step_x+1,step_y+1,step_z] == True and
|
|
||||||
# isNode[step_x,step_y,step_z+1] == True and isNode[step_x+1,step_y,step_z+1] == True and
|
|
||||||
# isNode[step_x,step_y+1,step_z+1] == True and isNode[step_x+1,step_y+1,step_z+1] == True):
|
|
||||||
# # create the node
|
|
||||||
# node = EM_FHNode.makeFHNode(X=pos_x+deltaX/2.0, Y=pos_y+deltaY/2.0, Z=pos_z+deltaZ/2.0)
|
|
||||||
# # store it in the array
|
|
||||||
# nodes[step_x,step_y,step_z] = node
|
|
||||||
# pos_z = pos_z + deltaZ
|
|
||||||
# pos_y = pos_y + deltaY
|
|
||||||
# pos_x = pos_x + deltaX
|
|
||||||
# # now create the grid of segments
|
|
||||||
# # first along x
|
|
||||||
# for step_z in range(0,stepsZ+1):
|
|
||||||
# for step_y in range(0,stepsY+1):
|
|
||||||
# for step_x in range(0,stepsX):
|
|
||||||
# # if the node and the next are inside the object shape, create the segment
|
|
||||||
# if nodes[step_x,step_y,step_z] <> None and nodes[step_x+1,step_y,step_z] <> None:
|
|
||||||
# segment = EM_FHSegment.makeFHSegment(nodeStart=nodes[step_x,step_y,step_z],nodeEnd=nodes[step_x+1,step_y,step_z],width=deltaX,height=deltaZ)
|
|
||||||
# # then along y
|
|
||||||
# for step_z in range(0,stepsZ+1):
|
|
||||||
# for step_x in range(0,stepsX+1):
|
|
||||||
# for step_y in range(0,stepsY):
|
|
||||||
# # if the node and the next are inside the object shape, create the segment
|
|
||||||
# if nodes[step_x,step_y,step_z] <> None and nodes[step_x,step_y+1,step_z] <> None:
|
|
||||||
# segment = EM_FHSegment.makeFHSegment(nodeStart=nodes[step_x,step_y,step_z],nodeEnd=nodes[step_x,step_y+1,step_z],width=deltaY,height=deltaZ)
|
|
||||||
# # finally along z
|
|
||||||
# for step_x in range(0,stepsX+1):
|
|
||||||
# for step_y in range(0,stepsY+1):
|
|
||||||
# for step_z in range(0,stepsZ):
|
|
||||||
# # if the node and the next are inside the object shape, create the segment
|
|
||||||
# if nodes[step_x,step_y,step_z] <> None and nodes[step_x,step_y,step_z+1] <> None:
|
|
||||||
# segment = EM_FHSegment.makeFHSegment(nodeStart=nodes[step_x,step_y,step_z],nodeEnd=nodes[step_x,step_y,step_z+1],width=deltaX,height=deltaY)
|
|
||||||
#
|
|
||||||
|
|
||||||
|
def getContainingBBox(objs):
|
||||||
|
''' Get the bounding box containing all the listed objects
|
||||||
|
|
||||||
|
'objs' is the list of FreeCAD objects
|
||||||
|
|
||||||
|
Returns the global bounding box.
|
||||||
|
If the list is None, or is not a list, or if the object have no Shape,
|
||||||
|
the returned BoundBox is None
|
||||||
|
'''
|
||||||
|
# create an empty bbox
|
||||||
|
gbbox = None
|
||||||
|
isfirst = True
|
||||||
|
# if 'objs' is not None
|
||||||
|
if objs:
|
||||||
|
if isinstance(objs,list):
|
||||||
|
for obj in objs:
|
||||||
|
if hasattr(obj,"Shape"):
|
||||||
|
if isfirst:
|
||||||
|
gbbox = obj.Shape.BoundBox
|
||||||
|
isfirst = False
|
||||||
|
else:
|
||||||
|
gbbox.add(obj.Shape.BoundBox)
|
||||||
|
return gbbox
|
||||||
|
|
||||||
|
def createVoxelSpace(bbox,delta):
|
||||||
|
''' Creates the voxel tensor (3D array) in the given bounding box
|
||||||
|
|
||||||
|
'bbox' is the overall FreeCAD.BoundBox bounding box
|
||||||
|
'delta' is the voxels size length
|
||||||
|
|
||||||
|
Returns a voxel tensor as a Numpy 3D array.
|
||||||
|
If gbbox is None, returns None
|
||||||
|
'''
|
||||||
|
if bbox == None:
|
||||||
|
return None
|
||||||
|
if delta == None:
|
||||||
|
return None
|
||||||
|
# add 1.0 to always cover the bbox space with the voxels
|
||||||
|
stepsX = int(bbox.XLength/delta + 1.0)
|
||||||
|
stepsY = int(bbox.YLength/delta + 1.0)
|
||||||
|
stepsZ = int(bbox.ZLength/delta + 1.0)
|
||||||
|
# debug
|
||||||
|
print("X="+str(stepsX)+" Y="+str(stepsY)+" Z="+str(stepsZ)+" tot="+str(stepsX*stepsY*stepsZ))
|
||||||
|
# create the 3D array of nodes as 16-bit integers (max 65k different conductivities)
|
||||||
|
voxelSpace=np.full((stepsX+1,stepsY+1,stepsZ+1), 0, np.int16)
|
||||||
|
return voxelSpace
|
||||||
|
|
||||||
|
def voxelizeConductor(obj,condIndex,gbbox,delta,voxelSpace):
|
||||||
|
''' Voxelize a solid object. The function will modify the 'voxelSpace'
|
||||||
|
by marking with 'condIndex' all the voxels that sample the object
|
||||||
|
'obj' internal.
|
||||||
|
|
||||||
|
'obj' is the object to voxelize
|
||||||
|
'condIndex' (integer) is the index of the object. It defines the object conductivity.
|
||||||
|
'gbbox' (FreeCAD.BoundBox) is the overall bounding box
|
||||||
|
'delta' is the voxels size length
|
||||||
|
'voxelSpace' (Numpy 3D array) is the voxel tensor of the overall space
|
||||||
|
'''
|
||||||
|
if obj == None:
|
||||||
|
return
|
||||||
|
if not hasattr(obj,"Shape"):
|
||||||
|
return
|
||||||
|
# get this object bbox
|
||||||
|
bbox = obj.Shape.BoundBox
|
||||||
|
# now must find the voxel set that contains the object bounding box
|
||||||
|
# find the voxel that contains the bbox min point
|
||||||
|
min_x = int((bbox.XMin - gbbox.XMin)/delta)
|
||||||
|
min_y = int((bbox.YMin - gbbox.YMin)/delta)
|
||||||
|
min_z = int((bbox.ZMin - gbbox.ZMin)/delta)
|
||||||
|
# find the voxel that contains the bbox max point
|
||||||
|
max_x = int((bbox.XMax - gbbox.XMin)/delta)
|
||||||
|
max_y = int((bbox.YMax - gbbox.YMin)/delta)
|
||||||
|
max_z = int((bbox.ZMax - gbbox.ZMin)/delta)
|
||||||
|
# and now iterate to find which voxel is inside the object 'obj',
|
||||||
|
# sampling based on the voxel centers
|
||||||
|
pos_x = gbbox.XMin + min_x * delta + delta/2.0
|
||||||
|
for step_x in range(min_x,max_x+1):
|
||||||
|
pos_y = gbbox.YMin + min_y * delta + delta/2.0
|
||||||
|
for step_y in range(min_y,max_y+1):
|
||||||
|
pos_z = gbbox.ZMin + min_z * delta + delta/2.0
|
||||||
|
for step_z in range(min_z,max_z+1):
|
||||||
|
# if the point is inside the object shape, or on the surface, flag it
|
||||||
|
if obj.Shape.isInside(Vector(pos_x,pos_y,pos_z),0.0,True):
|
||||||
|
# debug
|
||||||
|
#print("pos_x="+str(pos_x)+" pos_y="+str(pos_y)+" pos_z="+str(pos_z))
|
||||||
|
voxelSpace[step_x,step_y,step_z] = condIndex
|
||||||
|
pos_z = pos_z + delta
|
||||||
|
pos_y = pos_y + delta
|
||||||
|
pos_x = pos_x + delta
|
||||||
|
|
||||||
|
def createVoxelShell(obj,condIndex,gbbox,delta,voxelSpace=None):
|
||||||
|
''' Creates a shell composed by the external faces of a voxelized object.
|
||||||
|
|
||||||
|
'obj' is the object whose shell must be created
|
||||||
|
'condIndex' (integer) is the index of the object. It defines the object conductivity.
|
||||||
|
'gbbox' (FreeCAD.BoundBox) is the overall bounding box
|
||||||
|
'delta' is the voxels size length
|
||||||
|
'voxelSpace' (Numpy 3D array) is the voxel tensor of the overall space
|
||||||
|
'''
|
||||||
|
if voxelSpace == None:
|
||||||
|
return
|
||||||
|
if not hasattr(obj,"Shape"):
|
||||||
|
return
|
||||||
|
surfList = []
|
||||||
|
# get the object's bbox
|
||||||
|
bbox = obj.Shape.BoundBox
|
||||||
|
# now must find the voxel set that contains the object bounding box
|
||||||
|
# find the voxel that contains the bbox min point
|
||||||
|
min_x = int((bbox.XMin - gbbox.XMin)/delta)
|
||||||
|
min_y = int((bbox.YMin - gbbox.YMin)/delta)
|
||||||
|
min_z = int((bbox.ZMin - gbbox.ZMin)/delta)
|
||||||
|
# find the voxel that contains the bbox max point
|
||||||
|
max_x = int((bbox.XMax - gbbox.XMin)/delta)
|
||||||
|
max_y = int((bbox.YMax - gbbox.YMin)/delta)
|
||||||
|
max_z = int((bbox.ZMax - gbbox.ZMin)/delta)
|
||||||
|
# this is half the side of the voxel
|
||||||
|
halfdelta = delta/2.0
|
||||||
|
# array to find the six neighbour
|
||||||
|
sides = [(1,0,0), (-1,0,0), (0,1,0), (0,-1,0), (0,0,1), (0,0,-1)]
|
||||||
|
# vertexes of the six faces
|
||||||
|
vertexes = [[Vector(delta,0,0), Vector(delta,delta,0), Vector(delta,delta,delta), Vector(delta,0,delta)],
|
||||||
|
[Vector(0,0,0), Vector(0,0,delta), Vector(0,delta,delta), Vector(0,delta,0)],
|
||||||
|
[Vector(0,delta,0), Vector(0,delta,delta), Vector(delta,delta,delta), Vector(delta,delta,0)],
|
||||||
|
[Vector(0,0,0), Vector(delta,0,0), Vector(delta,0,delta), Vector(0,0,delta)],
|
||||||
|
[Vector(0,0,delta), Vector(delta,0,delta), Vector(delta,delta,delta), Vector(0,delta,delta)],
|
||||||
|
[Vector(0,0,0), Vector(0,delta,0), Vector(delta,delta,0), Vector(delta,0,0)]]
|
||||||
|
# and now iterate to find which voxel is inside the object 'obj',
|
||||||
|
# sampling based on the voxel centers
|
||||||
|
vbase = Vector(gbbox.XMin + min_x * delta, gbbox.YMin + min_y * delta, gbbox.ZMin + min_z * delta)
|
||||||
|
for step_x in range(min_x,max_x+1):
|
||||||
|
vbase.y = gbbox.YMin + min_y * delta
|
||||||
|
for step_y in range(min_y,max_y+1):
|
||||||
|
vbase.z = gbbox.ZMin + min_z * delta
|
||||||
|
for step_z in range(min_z,max_z+1):
|
||||||
|
# check if voxel is belonging to the given object
|
||||||
|
if voxelSpace[step_x,step_y,step_z] == condIndex:
|
||||||
|
# scan the six neighbour voxels, to see if they are belonging to the same conductor or not.
|
||||||
|
# If they are not belonging to the same conductor, or if the voxel space is finished, the current voxel
|
||||||
|
# side in the direction of the empty voxel is an external surface
|
||||||
|
for side, vertex in zip(sides,vertexes):
|
||||||
|
is_surface = False
|
||||||
|
nextVoxelIndexes = [step_x+side[0],step_y+side[1],step_z+side[2]]
|
||||||
|
if (nextVoxelIndexes[0] > max_x or nextVoxelIndexes[0] < 0 or
|
||||||
|
nextVoxelIndexes[1] > max_y or nextVoxelIndexes[1] < 0 or
|
||||||
|
nextVoxelIndexes[2] > max_z or nextVoxelIndexes[2] < 0):
|
||||||
|
is_surface = True
|
||||||
|
else:
|
||||||
|
if voxelSpace[nextVoxelIndexes[0],nextVoxelIndexes[1],nextVoxelIndexes[2]] != condIndex:
|
||||||
|
is_surface = True
|
||||||
|
if is_surface == True:
|
||||||
|
# debug
|
||||||
|
#print("pos_x="+str(vbase.x)+" pos_y="+str(vbase.y)+" pos_z="+str(vbase.z))
|
||||||
|
# create the face
|
||||||
|
# calculate the vertexes
|
||||||
|
v11 = vbase + vertex[0]
|
||||||
|
v12 = vbase + vertex[1]
|
||||||
|
v13 = vbase + vertex[2]
|
||||||
|
v14 = vbase + vertex[3]
|
||||||
|
# now make the face
|
||||||
|
poly = Part.makePolygon( [v11,v12,v13,v14,v11])
|
||||||
|
face = Part.Face(poly)
|
||||||
|
surfList.append(face)
|
||||||
|
vbase.z += delta
|
||||||
|
vbase.y += delta
|
||||||
|
vbase.x += delta
|
||||||
|
# create a shell. Does not need to be solid.
|
||||||
|
objShell = Part.makeShell(surfList)
|
||||||
|
return objShell
|
||||||
|
|
||||||
|
def findContactVoxelSurfaces(face,condIndex,gbbox,delta,voxelSpace=None,createShell=False):
|
||||||
|
''' Find the voxel surface sides corresponding to the given contact surface
|
||||||
|
(face) of an object. The object must have already been voxelized.
|
||||||
|
|
||||||
|
'face' is the object face
|
||||||
|
'condIndex' (integer) is the index of the object to which the face belongs.
|
||||||
|
It defines the object conductivity.
|
||||||
|
'gbbox' (FreeCAD.BoundBox) is the overall bounding box
|
||||||
|
'delta' is the voxels size length
|
||||||
|
'voxelSpace' (Numpy 3D array) is the voxel tensor of the overall space
|
||||||
|
'createShell' (bool) creates a shell out of the contact faces
|
||||||
|
|
||||||
|
Returns a list of surfaces in the format [x,y,z,voxside] where
|
||||||
|
x, y, z are the voxel position indexes, while voxside is '+x', '-x',
|
||||||
|
'+y', '-y', '+z', '-z' according the the impacted surface of the voxel
|
||||||
|
'''
|
||||||
|
if voxelSpace == None:
|
||||||
|
return
|
||||||
|
surfList = []
|
||||||
|
contactList = []
|
||||||
|
# get the face's bbox
|
||||||
|
bbox = face.BoundBox
|
||||||
|
# now must find the voxel set that contains the face bounding box
|
||||||
|
# with a certain slack - it could be the next voxel,
|
||||||
|
# if the surface is at the boundary between voxels.
|
||||||
|
# Find the voxel that contains the bbox min point
|
||||||
|
min_x = int((bbox.XMin - gbbox.XMin)/delta)-1
|
||||||
|
min_y = int((bbox.YMin - gbbox.YMin)/delta)-1
|
||||||
|
min_z = int((bbox.ZMin - gbbox.ZMin)/delta)-1
|
||||||
|
# find the voxel that contains the bbox max point
|
||||||
|
max_x = int((bbox.XMax - gbbox.XMin)/delta)+1
|
||||||
|
max_y = int((bbox.YMax - gbbox.YMin)/delta)+1
|
||||||
|
max_z = int((bbox.ZMax - gbbox.ZMin)/delta)+1
|
||||||
|
# debug
|
||||||
|
#print(str(min_x)+" "+str(min_y)+" "+str(min_z)+" "+str(max_x)+" "+str(max_y)+" "+str(max_z))
|
||||||
|
# create a Part.Vertex that we can use to test the distance
|
||||||
|
# to the face (as it is a TopoShape)
|
||||||
|
vec = FreeCAD.Vector(0,0,0)
|
||||||
|
testVertex = Part.Vertex(vec)
|
||||||
|
# this is half the side of the voxel
|
||||||
|
halfdelta = delta/2.0
|
||||||
|
# small displacement w.r.t. delta
|
||||||
|
epsdelta = delta/100.0
|
||||||
|
# array to find the six neighbour
|
||||||
|
sides = [(1,0,0), (-1,0,0), (0,1,0), (0,-1,0), (0,0,1), (0,0,-1)]
|
||||||
|
# string describing the side
|
||||||
|
sideStrs = ['+x', '-x', '+y', '-y', '+z', '-z']
|
||||||
|
# centers of the sides, with respect to the lower corner (with the smallest coordinates)
|
||||||
|
sideCenters = [Vector(delta,halfdelta,halfdelta), Vector(0.0,halfdelta,halfdelta),
|
||||||
|
Vector(halfdelta,delta,halfdelta), Vector(halfdelta,0.0,halfdelta),
|
||||||
|
Vector(halfdelta,halfdelta,delta), Vector(halfdelta,halfdelta,0.0)]
|
||||||
|
# vertexes of the six faces (with a slight offset)
|
||||||
|
vertexes = [[Vector(delta+epsdelta,0,0), Vector(delta+epsdelta,delta,0), Vector(delta+epsdelta,delta,delta), Vector(delta+epsdelta,0,delta)],
|
||||||
|
[Vector(-epsdelta,0,0), Vector(-epsdelta,0,delta), Vector(-epsdelta,delta,delta), Vector(-epsdelta,delta,0)],
|
||||||
|
[Vector(0,delta+epsdelta,0), Vector(0,delta+epsdelta,delta), Vector(delta,delta+epsdelta,delta), Vector(delta,delta+epsdelta,0)],
|
||||||
|
[Vector(0,-epsdelta,0), Vector(delta,-epsdelta,0), Vector(delta,-epsdelta,delta), Vector(0,-epsdelta,delta)],
|
||||||
|
[Vector(0,0,delta+epsdelta), Vector(delta,0,delta+epsdelta), Vector(delta,delta,delta+epsdelta), Vector(0,delta,delta+epsdelta)],
|
||||||
|
[Vector(0,0,-epsdelta), Vector(0,delta,-epsdelta), Vector(delta,delta,-epsdelta), Vector(delta,0,-epsdelta)]]
|
||||||
|
|
||||||
|
# and now iterate to find which voxel is inside the bounding box of the 'face',
|
||||||
|
vbase = Vector(gbbox.XMin + min_x * delta, gbbox.YMin + min_y * delta, gbbox.ZMin + min_z * delta)
|
||||||
|
for step_x in range(min_x,max_x+1):
|
||||||
|
vbase.y = gbbox.YMin + min_y * delta
|
||||||
|
for step_y in range(min_y,max_y+1):
|
||||||
|
vbase.z = gbbox.ZMin + min_z * delta
|
||||||
|
for step_z in range(min_z,max_z+1):
|
||||||
|
# check if voxel is belonging to the given object
|
||||||
|
if voxelSpace[step_x,step_y,step_z] == condIndex:
|
||||||
|
# scan the six neighbour voxels, to see if they are belonging to the same conductor or not.
|
||||||
|
# If they are not belonging to the same conductor, or if the voxel space is finished, the current voxel
|
||||||
|
# side in the direction of the empty voxel is an external surface
|
||||||
|
for side, sideStr, sideCenter, vertex in zip(sides,sideStrs,sideCenters,vertexes):
|
||||||
|
is_surface = False
|
||||||
|
nextVoxelIndexes = [step_x+side[0],step_y+side[1],step_z+side[2]]
|
||||||
|
if (nextVoxelIndexes[0] > max_x or nextVoxelIndexes[0] < 0 or
|
||||||
|
nextVoxelIndexes[1] > max_y or nextVoxelIndexes[1] < 0 or
|
||||||
|
nextVoxelIndexes[2] > max_z or nextVoxelIndexes[2] < 0):
|
||||||
|
is_surface = True
|
||||||
|
else:
|
||||||
|
if voxelSpace[nextVoxelIndexes[0],nextVoxelIndexes[1],nextVoxelIndexes[2]] != condIndex:
|
||||||
|
is_surface = True
|
||||||
|
if is_surface == True:
|
||||||
|
# debug
|
||||||
|
#print("pos_x="+str(vbase.x)+" pos_y="+str(vbase.y)+" pos_z="+str(vbase.z))
|
||||||
|
testVertex.Placement.Base = vbase + sideCenter
|
||||||
|
# if the point is close enough to the face, we consider
|
||||||
|
# the voxel surface as belonging to the voxelized face
|
||||||
|
dist = testVertex.distToShape(face)
|
||||||
|
# debug
|
||||||
|
#print(str(dist))
|
||||||
|
if abs(dist[0]) < halfdelta:
|
||||||
|
contactList.append([step_x,step_y,step_z,sideStr])
|
||||||
|
if createShell:
|
||||||
|
# create the face
|
||||||
|
# calculate the vertexes
|
||||||
|
v11 = vbase + vertex[0]
|
||||||
|
v12 = vbase + vertex[1]
|
||||||
|
v13 = vbase + vertex[2]
|
||||||
|
v14 = vbase + vertex[3]
|
||||||
|
# now make the face
|
||||||
|
poly = Part.makePolygon( [v11,v12,v13,v14,v11])
|
||||||
|
contFace = Part.Face(poly)
|
||||||
|
surfList.append(contFace)
|
||||||
|
vbase.z += delta
|
||||||
|
vbase.y += delta
|
||||||
|
vbase.x += delta
|
||||||
|
contactShell = None
|
||||||
|
if createShell:
|
||||||
|
if surfList != []:
|
||||||
|
# create a shell. Does not need to be solid.
|
||||||
|
contactShell = Part.makeShell(surfList)
|
||||||
|
return [contactList,contactShell]
|
||||||
|
|
||||||
#bb = App.BoundBox();
|
#bb = App.BoundBox();
|
||||||
#
|
#
|
||||||
|
|
|
@ -180,7 +180,7 @@ def parse_3D_input_file(fileinname, parentFid, parentFilePosMap, isdiel = False,
|
||||||
|
|
||||||
# build the file map (for single input file)
|
# build the file map (for single input file)
|
||||||
ret = create_file_map(fileinname, fid, filePosMap)
|
ret = create_file_map(fileinname, fid, filePosMap)
|
||||||
if ret <> True:
|
if ret != True:
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
panelVertexes = []
|
panelVertexes = []
|
||||||
|
@ -223,14 +223,14 @@ def parse_3D_input_file(fileinname, parentFid, parentFilePosMap, isdiel = False,
|
||||||
# conductor name because in the same file called more than once)
|
# conductor name because in the same file called more than once)
|
||||||
if m_iParseLevel == 0:
|
if m_iParseLevel == 0:
|
||||||
localGroupname = "g"
|
localGroupname = "g"
|
||||||
else
|
else:
|
||||||
localGroupname = groupname
|
localGroupname = groupname
|
||||||
localGroupname = localGroupname + str(m_iGroupNum[m_iParseLevel]) + '_'
|
localGroupname = localGroupname + str(m_iGroupNum[m_iParseLevel]) + '_'
|
||||||
|
|
||||||
# read optional values
|
# read optional values
|
||||||
if len(splitLine) >= 7:
|
if len(splitLine) >= 7:
|
||||||
# read optional '+'. If not a '+', increment the group
|
# read optional '+'. If not a '+', increment the group
|
||||||
if splitLine[6] <> '+':
|
if splitLine[6] != '+':
|
||||||
# increase group name
|
# increase group name
|
||||||
m_iGroupNum[m_iParseLevel] = m_iGroupNum[m_iParseLevel] + 1
|
m_iGroupNum[m_iParseLevel] = m_iGroupNum[m_iParseLevel] + 1
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user