forked from psicofil/SalomeToCalculix
-
Notifications
You must be signed in to change notification settings - Fork 1
/
salome2ccx.py
230 lines (179 loc) · 7.61 KB
/
salome2ccx.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
# -*- coding: utf-8 -*-
"""
© Lucio Gomez, April 2016 - original code
© Ihor Mirzov, August 2019 - refactoring and improvements
Distributed under GNU General Public License v3.0
Exports Salome mesh to Calculix INP format.
Run from Salome Mesh module with File->Load Script... (Ctrl+T)
Before run update variables 'cgx_bin' and 'script_dir' below.
"""
# Path to CGX binary (exapmle for Linux)
cgx_bin = '/usr/local/bin/cgx'
# Path to the script folder
script_dir = '/media/ihor/WORK/Calculix/salome2ccx'
import os
import sys
import subprocess
import tempfile
import inspect
import shutil
try:
import salome
from salome.smesh import smeshBuilder
except:
print('Run this script from SALOME.')
sys.exit()
try:
from PyQt4 import QtGui, QtCore, uic
from PyQt4.QtGui import *
from PyQt4.QtCore import *
except:
from PyQt5 import QtCore, QtGui, uic
from PyQt5.QtWidgets import *
class Dialog(QDialog):
def __init__(self):
# current_script = inspect.getfile(inspect.currentframe()) # full path to this script
# script_dir = os.path.dirname(current_script) # path to directory with this script
# Create dialog window
super(Dialog, self).__init__()
# Load form for GUI
gui_form_file = os.path.join(script_dir, 'salome2ccx.ui')
uic.loadUi(gui_form_file, self)
self.meshes = []
# Actions
self.pb_sel_mesh.clicked.connect(self.select_meshes)
self.pb_output_folder.clicked.connect(self.output_folder)
self.okbox.button(QDialogButtonBox.Reset).clicked.connect(self.reset)
self.okbox.rejected.connect(lambda: self.reject()) # close window
self.okbox.accepted.connect(self.convert)
# Clear selected mesh list
def reset(self):
self.le_selectMesh.setText('')
self.meshes = []
self.le_output_folder.setText('')
# Select folder to put conversion results to
def output_folder(self):
INP_dir = QFileDialog.getExistingDirectory(self,
'Select output folder', script_dir,
QFileDialog.ShowDirsOnly)
self.le_output_folder.setText(INP_dir)
# Open CGX to view conversion result
def open_CGX(self, inp_full_path):
try:
command_cgx = cgx_bin + ' -c ' + inp_full_path
QtCore.QProcess().startDetached(command_cgx)
except:
QMessageBox.critical(None,
'Abort!',
'Can\'t open result with CGX.',
QMessageBox.Abort)
# Perform conversion
def convert(self):
success = True
# If meshes are chosen
if len(self.meshes):
# Chosen output folder
output_folder = self.le_output_folder.text()
if os.path.isdir(output_folder):
# Path to chosen converter's binary
extension = ('.exe' if os.name=='nt' else '') # file extension in OS
binary = os.path.join(script_dir, 'converters',
self.converter.currentText() + extension)
# For each of selected meshes
for mesh in self.meshes:
if self.rb_delet_e_f.isChecked():
self.delete_edges_and_faces(mesh)
# UNV file name is a mesh name
unv_file_name = mesh.GetName() # file name only
if unv_file_name.endswith('.unv'):
unv_file_name = unv_file_name[:-4]
# Export UNV file into TEMP folder: path can't be too long - Salome BUG
temp_folder = os.path.dirname(tempfile.mkstemp()[1])
unv_full_path = os.path.join(temp_folder, unv_file_name) # file name with path
mesh.ExportUNV(unv_full_path + '.unv')
try:
# Perform conversion
if 'unical3' in binary:
command = ' '.join([binary, unv_full_path]) # without extension
output = subprocess.check_output([command, '-1'],
shell=True, stderr=subprocess.STDOUT)
else: # unv2ccx
command = ' '.join([binary, unv_full_path + '.unv']) # with extension
output = subprocess.check_output(command,
shell=True, stderr=subprocess.STDOUT)
# Move result file from TEMP to output directory
inp_full_path = os.path.join(output_folder, unv_file_name) + '.inp'
shutil.move(unv_full_path + '.inp', inp_full_path)
except:
success = False
QMessageBox.critical(None,
'Error!',
'Can\'t convert ' + unv_file_name,
QMessageBox.Ok)
# Delete temp file
os.remove(unv_full_path + '.unv')
# For each mesh open CGX to view conversion result
if self.rb_cgx.isChecked():
self.open_CGX(inp_full_path)
# Result message box
if success:
QMessageBox.information(None,
'Ok!',
'Operation successfull!',
QMessageBox.Ok)
# Exit
self.accept()
else:
QMessageBox.warning(None,
'Warning!',
'Please, select output folder!',
QMessageBox.Ok)
else:
QMessageBox.warning(None,
'Warning!',
'Please, select meshes!',
QMessageBox.Ok)
# Delete_edges_and_faces from mesh
# TODO Doesn't work!
# def delete_edges_and_faces(self, mesh):
# Group_1 = mesh.CreateEmptyGroup(SMESH.FACE, 'Group_1' )
# nbAdd = Group_1.AddFrom( mesh.GetMesh() )
# Group_2 = mesh.CreateEmptyGroup( SMESH.EDGE, 'Group_2' )
# nbAdd = Group_2.AddFrom( mesh.GetMesh() )
# mesh.RemoveGroupWithContents(Group_1)
# mesh.RemoveGroupWithContents(Group_2)
# Get list of selected meshes
def select_meshes(self):
smesh = smeshBuilder.New(salome.myStudy)
# Get selected meshes
selCount = salome.sg.SelectedCount() # total number of selected items
for i in range(selCount):
selected = salome.sg.getSelected(i)
selobjID = salome.myStudy.FindObjectID(selected)
selobj = selobjID.GetObject()
mName = selobjID.GetName().replace(' ','_')
try:
mesh = smesh.Mesh(selobj)
self.meshes.append(mesh)
mesh_name = mesh.GetName()
if not mesh_name in self.le_selectMesh.toPlainText():
self.le_selectMesh.append(mesh_name)
except:
selCount = 0
break
# Wrong selection or nothing selected
if not selCount:
QMessageBox.critical(None,
'Abort!',
'Please, select meshes in the Object Browser. Only meshes.',
QMessageBox.Abort)
if __name__ == '__main__':
if not os.path.isfile(cgx_bin):
print('ERROR! Update \"cgx_bin\" variable in the code.')
sys.exit()
if not os.path.isdir(script_dir):
print('ERROR! Update \"script_dir\" variable in the code.')
sys.exit()
if salome.sg.hasDesktop():
salome.sg.updateObjBrowser()
Dialog().show()