464 lines
20 KiB
HTML
464 lines
20 KiB
HTML
<html><head><title>Embedding FreeCADGui/cs</title><meta http-equiv="Content-Type" content="text/html; charset=utf-8"><link type='text/css' href='wiki.css' rel='stylesheet'></head><body><h1>Embedding FreeCADGui/cs</h1></div>
|
|
|
|
<div id="mw-content-text" lang="cs" dir="ltr" class="mw-content-ltr"><hr/><div class="mw-parser-output"><p>You already know that you can <a href="Embedding_FreeCAD.html" title="Embedding FreeCAD">import the FreeCAD module</a> into a python application, and use all its tools from the host application. But the FreeCAD User Interface (GUI) can also be imported as a python module. Normally you can only import the complete interface as a whole, not pieces of it. That is because the FreeCAD interface system is not only made of independent widgets and toolbars, but is a complex construction where several invisible components (such as the selection system, etc) are needed for the main 3D view to be able to function.
|
|
</p><p>But, with a bit of hacking, it is possible to import the whole FreeCAD interface, then move the 3D view from it to your own Qt application. We show here 3 different methods.
|
|
</p>
|
|
<div id="toc" class="toc"><div class="toctitle"><h2>Contents</h2></div>
|
|
<ul>
|
|
<li class="toclevel-1 tocsection-1"><a href="#Using_the_FreeCAD_3D_view_widget_directly"><span class="tocnumber">1</span> <span class="toctext">Using the FreeCAD 3D view widget directly</span></a></li>
|
|
<li class="toclevel-1 tocsection-2"><a href="#Creating_a_soGui_Examiner_Viewer"><span class="tocnumber">2</span> <span class="toctext">Creating a soGui Examiner Viewer</span></a></li>
|
|
<li class="toclevel-1 tocsection-3"><a href="#Using_the_quarter_module"><span class="tocnumber">3</span> <span class="toctext">Using the quarter module</span></a></li>
|
|
<li class="toclevel-1 tocsection-4"><a href="#Without_even_firing_up_the_FreeCAD_Gui"><span class="tocnumber">4</span> <span class="toctext">Without even firing up the FreeCAD Gui</span></a></li>
|
|
</ul>
|
|
</div>
|
|
|
|
<h2><span class="mw-headline" id="Using_the_FreeCAD_3D_view_widget_directly">Using the FreeCAD 3D view widget directly</span></h2>
|
|
<p>Be aware that there are a lot of problems with this approach. The Qt event handling doesn't seem to work (no idea why) and if you use the 3d view's context-menu the application crashes. A better way could be to create your own 3d view SoQtExaminerViewer or SoQtViewer and "push" the content of FreeCAD's 3d view to your view, as shown in the other sections below.
|
|
</p><p>First, get the main window via PyQt:
|
|
</p>
|
|
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>from PySide import QtGui
|
|
from PySide import QtCore
|
|
|
|
def getMainWindow():
|
|
toplevel = QtGui.qApp.topLevelWidgets()
|
|
for i in toplevel:
|
|
if i.metaObject().className() == "Gui::MainWindow":
|
|
return i
|
|
raise Exception("No main window found")
|
|
|
|
mw=getMainWindow()</pre></div>
|
|
<p>Then get the View3DInventor view the same way:
|
|
</p>
|
|
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>def get3dview(mw):
|
|
childs=mw.findChildren(QtGui.QMainWindow)
|
|
for i in childs:
|
|
if i.metaObject().className()=="Gui::View3DInventor":
|
|
return i
|
|
return None
|
|
|
|
v=get3dview(mw)</pre></div>
|
|
<p>The following code is generated automatically, by <a href="Dialog_creation.html" title="Dialog creation">creating a Ui-file with QtDesigner</a>, and converting it to python code with the pyuic tool:
|
|
</p>
|
|
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># -*- coding: utf-8 -*-
|
|
|
|
# Form implementation generated from reading ui file 'mainwindow.ui'
|
|
#
|
|
# Created: Sun Dec 27 11:18:56 2009
|
|
# by: PySide UI code generator 4.6
|
|
#
|
|
# Modify for PySide 11/02/2015
|
|
# Python version: 2.7.8
|
|
# Qt version: 4.8.6
|
|
#
|
|
# WARNING! All changes made in this file will be lost!
|
|
|
|
from PySide import QtCore, QtGui
|
|
|
|
class Ui_MainWindow(object):
|
|
def setupUi(self, MainWindow):
|
|
MainWindow.setObjectName("MainWindow")
|
|
MainWindow.resize(508, 436)
|
|
self.centralwidget = QtGui.QWidget(MainWindow)
|
|
self.centralwidget.setObjectName("centralwidget")
|
|
self.gridLayout = QtGui.QGridLayout(self.centralwidget)
|
|
self.gridLayout.setObjectName("gridLayout")
|
|
self.mdiArea = QtGui.QMdiArea(self.centralwidget)
|
|
self.mdiArea.setViewMode(QtGui.QMdiArea.TabbedView)
|
|
self.mdiArea.setTabPosition(QtGui.QTabWidget.South)
|
|
self.mdiArea.setObjectName("mdiArea")
|
|
self.gridLayout.addWidget(self.mdiArea, 0, 0, 1, 1)
|
|
MainWindow.setCentralWidget(self.centralwidget)
|
|
self.menubar = QtGui.QMenuBar(MainWindow)
|
|
self.menubar.setGeometry(QtCore.QRect(0, 0, 508, 27))
|
|
self.menubar.setObjectName("menubar")
|
|
MainWindow.setMenuBar(self.menubar)
|
|
self.statusbar = QtGui.QStatusBar(MainWindow)
|
|
self.statusbar.setObjectName("statusbar")
|
|
MainWindow.setStatusBar(self.statusbar)
|
|
|
|
self.retranslateUi(MainWindow)
|
|
QtCore.QMetaObject.connectSlotsByName(MainWindow)
|
|
|
|
def retranslateUi(self, MainWindow):
|
|
MainWindow.setWindowTitle(QtGui.QApplication.translate("MainWindow", "MainWindow", None, QtGui.QApplication.UnicodeUTF8))</pre></div>
|
|
<p>Then, create a main window that should be your application's main window, apply the UI setup above to it in order to add an MDI area and "move" our 3d view to it
|
|
</p>
|
|
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>ui=Ui_MainWindow()
|
|
my_mw=QtGui.QMainWindow()
|
|
ui.setupUi(my_mw)
|
|
ui.mdiArea.addSubWindow(v)
|
|
my_mw.show()</pre></div>
|
|
<h2><span class="mw-headline" id="Creating_a_soGui_Examiner_Viewer">Creating a soGui Examiner Viewer</span></h2>
|
|
<p>Alternatively, you can also use the FreeCADGui module to extract a coin/openInventor representation of the objects of your scene, then use that coin data in an external viewer (your application). Here is an easy way to get the 3d representation of an object:
|
|
</p>
|
|
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre> FreeCAD.activeDocument().addObject("Part::Box","myBox")
|
|
s=FreeCADGui.activeDocument().getObject("myBox").toString() # store as string
|
|
from pivy import coin
|
|
inp.setBuffer(s)
|
|
myNode=coin.SoDB.readAll(inp) # restore from string</pre></div>
|
|
<p>Then, create a standalone viewer with pivy:
|
|
</p>
|
|
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>from pivy.sogui import *
|
|
from pivy.coin import *
|
|
import sys
|
|
|
|
def myViewer():
|
|
# Initialize Coin. This returns a main window to use.
|
|
# If unsuccessful, exit.
|
|
myWindow = SoGui.init(sys.argv[0])
|
|
if myWindow == None: sys.exit(1)
|
|
|
|
# Make an empty scene and add our node to it
|
|
scene = SoSeparator()
|
|
scene.addChild(myNode)
|
|
|
|
# Create a viewer in which to see our scene graph.
|
|
viewer = SoGuiExaminerViewer(myWindow)
|
|
|
|
# Put our scene into viewer, change the title
|
|
viewer.setSceneGraph(scene)
|
|
viewer.setTitle("FreeCAD Object Viewer")
|
|
viewer.show()
|
|
|
|
SoGui.show(myWindow) # Display main window
|
|
SoGui.mainLoop() # Main Coin event loop</pre></div>
|
|
<p>Then you just need to run your viewer:
|
|
</p>
|
|
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>myViewer()</pre></div>
|
|
<h2><span class="mw-headline" id="Using_the_quarter_module">Using the quarter module</span></h2>
|
|
<p>Instead of using the sogui viewer, you can also use the more modern quarter module. This is probably the best solution of the 3.
|
|
</p>
|
|
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre> #!/usr/bin/env python
|
|
|
|
###
|
|
# Copyright (c) 2002-2008 Kongsberg SIM
|
|
#
|
|
# Permission to use, copy, modify, and distribute this software for any
|
|
# purpose with or without fee is hereby granted, provided that the above
|
|
# copyright notice and this permission notice appear in all copies.
|
|
#
|
|
# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
#
|
|
|
|
import os
|
|
import sys
|
|
|
|
from PyQt4 import QtCore, QtGui
|
|
from PyQt4.QtGui import QMainWindow, QWorkspace, QAction, QFileDialog, QApplication
|
|
|
|
from pivy.coin import SoInput, SoDB
|
|
from pivy.quarter import QuarterWidget
|
|
|
|
import FreeCAD, FreeCADGui
|
|
|
|
def getMainWindow():
|
|
toplevel = QtGui.qApp.topLevelWidgets()
|
|
for i in toplevel:
|
|
if i.metaObject().className() == "Gui::MainWindow":
|
|
return i
|
|
raise Exception("No main window found")
|
|
|
|
class MdiQuarterWidget(QuarterWidget):
|
|
def __init__(self, parent, sharewidget):
|
|
QuarterWidget.__init__(self, parent=parent, sharewidget=sharewidget)
|
|
|
|
def loadFile(self, filename):
|
|
in_ = SoInput()
|
|
if (in_.openFile(str(filename.toLatin1()))):
|
|
root = SoDB.readAll(in_)
|
|
if (root):
|
|
self.setSceneGraph(root)
|
|
self.currentfile = filename
|
|
self.setWindowTitle(filename)
|
|
return True
|
|
return False
|
|
|
|
def currentFile(self):
|
|
return self.currentfile
|
|
|
|
def minimumSizeHint(self):
|
|
return QtCore.QSize(640, 480)
|
|
|
|
class MdiMainWindow(QMainWindow):
|
|
def __init__(self, qApp):
|
|
QMainWindow.__init__(self)
|
|
self._firstwidget = None
|
|
self._workspace = QWorkspace()
|
|
self.setCentralWidget(self._workspace)
|
|
self.setAcceptDrops(True)
|
|
self.setWindowTitle("Pivy Quarter MDI example")
|
|
|
|
filemenu = self.menuBar().addMenu("&File")
|
|
windowmenu = self.menuBar().addMenu("&Windows")
|
|
|
|
fileopenaction = QAction("&Create Box", self)
|
|
fileexitaction = QAction("E&xit", self)
|
|
tileaction = QAction("Tile", self)
|
|
cascadeaction = QAction("Cascade", self)
|
|
|
|
filemenu.addAction(fileopenaction)
|
|
filemenu.addAction(fileexitaction)
|
|
windowmenu.addAction(tileaction)
|
|
windowmenu.addAction(cascadeaction)
|
|
|
|
self.connect(fileopenaction, QtCore.SIGNAL("triggered()"), self.createBoxInFreeCAD)
|
|
self.connect(fileexitaction, QtCore.SIGNAL("triggered()"), QtGui.qApp.closeAllWindows)
|
|
self.connect(tileaction, QtCore.SIGNAL("triggered()"), self._workspace.tile)
|
|
self.connect(cascadeaction, QtCore.SIGNAL("triggered()"), self._workspace.cascade)
|
|
|
|
windowmapper = QtCore.QSignalMapper(self)
|
|
self.connect(windowmapper, QtCore.SIGNAL("mapped(QWidget *)"), self._workspace.setActiveWindow)
|
|
|
|
self.dirname = os.curdir
|
|
|
|
def dragEnterEvent(self, event):
|
|
# just accept anything...
|
|
event.acceptProposedAction()
|
|
|
|
def dropEvent(self, event):
|
|
mimedata = event.mimeData()
|
|
if mimedata.hasUrls():
|
|
path = mimedata.urls().takeFirst().path()
|
|
self.open_path(path)
|
|
|
|
def closeEvent(self, event):
|
|
self._workspace.closeAllWindows()
|
|
|
|
def open(self):
|
|
self.open_path(QFileDialog.getOpenFileName(self, "", self.dirname))
|
|
|
|
def open_path(self, filename):
|
|
self.dirname = os.path.dirname(str(filename.toLatin1()))
|
|
if not filename.isEmpty():
|
|
existing = self.findMdiChild(filename)
|
|
if existing:
|
|
self._workspace.setActiveWindow(existing)
|
|
return
|
|
child = self.createMdiChild()
|
|
if (child.loadFile(filename)):
|
|
self.statusBar().showMessage("File loaded", 2000)
|
|
child.show()
|
|
else:
|
|
child.close()
|
|
|
|
def findMdiChild(self, filename):
|
|
canonicalpath = QtCore.QFileInfo(filename).canonicalFilePath()
|
|
for window in self._workspace.windowList():
|
|
mdiwidget = window
|
|
if mdiwidget.currentFile() == canonicalpath:
|
|
return mdiwidget
|
|
return 0;
|
|
|
|
def createMdiChild(self):
|
|
widget = MdiQuarterWidget(None, self._firstwidget)
|
|
self._workspace.addWindow(widget)
|
|
if not self._firstwidget:
|
|
self._firstwidget = widget
|
|
return widget
|
|
|
|
def createBoxInFreeCAD(self):
|
|
widget = MdiQuarterWidget(None, self._firstwidget)
|
|
self._workspace.addWindow(widget)
|
|
if not self._firstwidget:
|
|
self._firstwidget = widget
|
|
widget.show()
|
|
doc = FreeCAD.newDocument()
|
|
doc.addObject("Part::Box","myBox")
|
|
iv_=FreeCADGui.getDocument(doc.Name).getObject("myBox").toString()
|
|
in_ = SoInput()
|
|
in_.setBuffer(iv_)
|
|
root = SoDB.readAll(in_)
|
|
if (root):
|
|
widget.setSceneGraph(root)
|
|
|
|
def main():
|
|
app = QApplication(sys.argv)
|
|
mdi = MdiMainWindow(app)
|
|
mdi.show()
|
|
FreeCADGui.showMainWindow() # setup the GUI stuff of FreeCAD
|
|
mw=getMainWindow()
|
|
mw.hide() # hide all
|
|
if len(sys.argv)==2:
|
|
mdi.open_path(QtCore.QString(sys.argv[1]))
|
|
sys.exit(app.exec_())
|
|
|
|
def show():
|
|
mdi = MdiMainWindow(QtGui.qApp)
|
|
mdi.show()
|
|
mw=getMainWindow()
|
|
#mw.hide() # hide all
|
|
|
|
if __name__ == '__main__':
|
|
main()</pre></div>
|
|
<h2><span class="mw-headline" id="Without_even_firing_up_the_FreeCAD_Gui">Without even firing up the FreeCAD Gui</span></h2>
|
|
<p>Starting from FreeCAD rev2760, it is now possible to obtain the coin representation of any FreeCAD object without opening the main window. This makes it extremely easy to implement your own viewer and transparently have FreeCAD updating it. After importing the FreeCADGui module, you need to fire it up with the setupWithoutGUI() method, after which you can use all of FreeCAD's view providers to obtain coin/openInventor nodes.
|
|
</p>
|
|
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre> import os, sys, FreeCAD, FreeCADGui
|
|
from PyQt4 import QtCore, QtGui
|
|
from PyQt4.QtGui import QMainWindow, QWorkspace, QAction, QFileDialog, QApplication
|
|
from pivy.coin import SoInput, SoDB, sogui
|
|
|
|
class MdiMainWindow(QMainWindow):
|
|
def __init__(self, qApp):
|
|
QMainWindow.__init__(self)
|
|
self._firstwidget = None
|
|
self._workspace = QWorkspace()
|
|
self.setCentralWidget(self._workspace)
|
|
self.setAcceptDrops(True)
|
|
self.setWindowTitle("Pivy Quarter MDI example")
|
|
self.viewers=[]
|
|
|
|
filemenu = self.menuBar().addMenu("&File")
|
|
windowmenu = self.menuBar().addMenu("&Windows")
|
|
|
|
fileopenaction = QAction("&Create Box", self)
|
|
fileexitaction = QAction("E&xit", self)
|
|
tileaction = QAction("Tile", self)
|
|
cascadeaction = QAction("Cascade", self)
|
|
|
|
filemenu.addAction(fileopenaction)
|
|
filemenu.addAction(fileexitaction)
|
|
windowmenu.addAction(tileaction)
|
|
windowmenu.addAction(cascadeaction)
|
|
|
|
self.connect(fileopenaction, QtCore.SIGNAL("triggered()"), self.createBoxInFreeCAD)
|
|
self.connect(fileexitaction, QtCore.SIGNAL("triggered()"), QtGui.qApp.closeAllWindows)
|
|
self.connect(tileaction, QtCore.SIGNAL("triggered()"), self._workspace.tile)
|
|
self.connect(cascadeaction, QtCore.SIGNAL("triggered()"), self._workspace.cascade)
|
|
|
|
windowmapper = QtCore.QSignalMapper(self)
|
|
self.connect(windowmapper, QtCore.SIGNAL("mapped(QWidget *)"), self._workspace.setActiveWindow)
|
|
|
|
def closeEvent(self, event):
|
|
self._workspace.closeAllWindows()
|
|
|
|
def createBoxInFreeCAD(self):
|
|
widget = QtGui.QWidget(self._firstwidget)
|
|
viewer = sogui.SoGuiExaminerViewer(widget)
|
|
self._workspace.addWindow(widget)
|
|
if not self._firstwidget:
|
|
self._firstwidget = widget
|
|
widget.show()
|
|
self.viewers.append(viewer)
|
|
doc = FreeCAD.newDocument()
|
|
obj=doc.addObject("Part::Box","myBox")
|
|
doc.recompute()
|
|
root=FreeCADGui.subgraphFromObject(obj)
|
|
viewer.setSceneGraph(root)
|
|
|
|
def main():
|
|
app = QApplication(sys.argv)
|
|
mdi = MdiMainWindow(app)
|
|
mdi.show()
|
|
FreeCADGui.setupWithoutGUI()
|
|
sys.exit(app.exec_())
|
|
|
|
if __name__ == '__main__':
|
|
main()</pre></div>
|
|
<p>Or, if using pivy's sogui module doesn't work for you (the sogui module is becoming obsoleted and the coin developers are now favoring the new quarter library, which has much better interaction with qt), this is the same script but with using quarter:
|
|
</p>
|
|
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre> #!/usr/bin/env python
|
|
|
|
import os
|
|
import sys
|
|
|
|
from PyQt4 import QtCore, QtGui
|
|
from PyQt4.QtGui import QMainWindow, QWorkspace, QAction, QApplication
|
|
|
|
from pivy.coin import SoInput, SoDB
|
|
from pivy.quarter import QuarterWidget
|
|
import FreeCADGui
|
|
|
|
|
|
class MdiQuarterWidget(QuarterWidget):
|
|
def __init__(self, parent, sharewidget):
|
|
QuarterWidget.__init__(self, parent=parent, sharewidget=sharewidget)
|
|
|
|
def minimumSizeHint(self):
|
|
return QtCore.QSize(640, 480)
|
|
|
|
|
|
class MdiMainWindow(QMainWindow):
|
|
def __init__(self, qApp):
|
|
QMainWindow.__init__(self)
|
|
self._firstwidget = None
|
|
self._workspace = QWorkspace()
|
|
self.setCentralWidget(self._workspace)
|
|
self.setAcceptDrops(True)
|
|
self.setWindowTitle("Pivy Quarter MDI example")
|
|
|
|
filemenu = self.menuBar().addMenu("&File")
|
|
windowmenu = self.menuBar().addMenu("&Windows")
|
|
|
|
fileopenaction = QAction("&Create Box", self)
|
|
fileexitaction = QAction("E&xit", self)
|
|
tileaction = QAction("Tile", self)
|
|
cascadeaction = QAction("Cascade", self)
|
|
|
|
filemenu.addAction(fileopenaction)
|
|
filemenu.addAction(fileexitaction)
|
|
windowmenu.addAction(tileaction)
|
|
windowmenu.addAction(cascadeaction)
|
|
|
|
self.connect(fileopenaction, QtCore.SIGNAL("triggered()"), self.createBoxInFreeCAD)
|
|
self.connect(fileexitaction, QtCore.SIGNAL("triggered()"), QtGui.qApp.closeAllWindows)
|
|
self.connect(tileaction, QtCore.SIGNAL("triggered()"), self._workspace.tile)
|
|
self.connect(cascadeaction, QtCore.SIGNAL("triggered()"), self._workspace.cascade)
|
|
|
|
windowmapper = QtCore.QSignalMapper(self)
|
|
self.connect(windowmapper, QtCore.SIGNAL("mapped(QWidget *)"), self._workspace.setActiveWindow)
|
|
|
|
self.dirname = os.curdir
|
|
|
|
def closeEvent(self, event):
|
|
self._workspace.closeAllWindows()
|
|
|
|
def createBoxInFreeCAD(self):
|
|
d=FreeCAD.newDocument()
|
|
o=d.addObject("Part::Box")
|
|
d.recompute()
|
|
s=FreeCADGui.subgraphFromObject(o)
|
|
child = self.createMdiChild()
|
|
child.show()
|
|
child.setSceneGraph(s)
|
|
|
|
def createMdiChild(self):
|
|
widget = MdiQuarterWidget(None, self._firstwidget)
|
|
self._workspace.addWindow(widget)
|
|
if not self._firstwidget:
|
|
self._firstwidget = widget
|
|
return widget
|
|
|
|
|
|
def main():
|
|
app = QApplication(sys.argv)
|
|
FreeCADGui.setupWithoutGUI()
|
|
mdi = MdiMainWindow(app)
|
|
mdi.show()
|
|
sys.exit(app.exec_())
|
|
|
|
|
|
if __name__ == '__main__':
|
|
main()</pre></div>
|
|
<div style="clear:both"></div>
|
|
</div>
|
|
|
|
|
|
|
|
</div>
|
|
|
|
</div><div class="printfooter">
|
|
Online version: "<a dir="ltr" href="https://www.freecadweb.org/wiki/index.php?title=Embedding_FreeCADGui/cs&oldid=160245">http://www.freecadweb.org/wiki/index.php?title=Embedding_FreeCADGui/cs&oldid=160245</a>"</div>
|
|
<div id="catlinks" class="catlinks" data-mw="interface"></div><div class="visualClear"></div>
|
|
</div>
|
|
</div>
|
|
<div id="mw-navigation">
|
|
<h2>Navigation menu</h2>
|
|
|
|
</body></html> |