Changeset 706bb4e in sasview for src/sas/sasgui
- Timestamp:
- Sep 18, 2017 10:20:29 AM (7 years ago)
- Branches:
- master, ESS_GUI, ESS_GUI_Docs, ESS_GUI_batch_fitting, ESS_GUI_bumps_abstraction, ESS_GUI_iss1116, ESS_GUI_iss879, ESS_GUI_iss959, ESS_GUI_opencl, ESS_GUI_ordering, ESS_GUI_sync_sascalc, magnetic_scatt, release-4.2.2, ticket-1009, ticket-1094-headless, ticket-1242-2d-resolution, ticket-1243, ticket-1249, ticket885, unittest-saveload
- Children:
- 60a7820
- Parents:
- 460d3a1 (diff), cfd27dd (diff)
Note: this is a merge changeset, the changes displayed below correspond to the merge itself.
Use the (diff) links above to see all the changes relative to each parent. - Location:
- src/sas/sasgui
- Files:
-
- 3 added
- 14 deleted
- 22 edited
Legend:
- Unmodified
- Added
- Removed
-
src/sas/sasgui/guiframe/local_perspectives/plotting/plotting.py
r235f514 r2d9526d 14 14 import wx 15 15 import sys 16 from copy import deepcopy 16 17 from sas.sasgui.guiframe.events import EVT_NEW_PLOT 17 18 from sas.sasgui.guiframe.events import EVT_PLOT_QRANGE … … 275 276 action_check = True 276 277 else: 278 if action_string == 'update': 279 # Update all existing plots of data with this ID 280 for data in event.plots: 281 for panel in self.plot_panels.values(): 282 if data.id in panel.plots.keys(): 283 plot_exists = True 284 # Pass each panel it's own copy of the data 285 # that's being updated, otherwise things like 286 # colour and line thickness are unintentionally 287 # synced across panels 288 self.update_panel(deepcopy(data), panel) 289 return 290 277 291 group_id = event.group_id 278 if group_id in self.plot_panels .keys():292 if group_id in self.plot_panels: 279 293 #remove data from panel 280 294 if action_string == 'remove': -
src/sas/sasgui/perspectives/calculator/model_editor.py
r07ec714 r23359ccb 106 106 self.model2_string = "cylinder" 107 107 self.name = 'Sum' + M_NAME 108 self.factor = 'scale_factor'109 108 self._notes = '' 110 109 self._operator = '+' … … 133 132 self.model2_name = str(self.model2.GetValue()) 134 133 self.good_name = True 135 self.fill_op rator_combox()134 self.fill_operator_combox() 136 135 137 136 def _layout_name(self): … … 491 490 a sum or multiply model then create the appropriate string 492 491 """ 493 494 492 name = '' 495 496 493 if operator == '*': 497 494 name = 'Multi' 498 factor = 'BackGround' 499 f_oper = '+' 495 factor = 'background' 500 496 else: 501 497 name = 'Sum' 502 498 factor = 'scale_factor' 503 f_oper = '*' 504 505 self.factor = factor 499 506 500 self._operator = operator 507 self.explanation = " Plugin Model = %s %s (model1 %s model2)\n" % \508 (self.factor, f_oper, self._operator)501 self.explanation = (" Plugin_model = scale_factor * (model_1 {} " 502 "model_2) + background").format(operator) 509 503 self.explanationctr.SetLabel(self.explanation) 510 504 self.name = name + M_NAME 511 505 512 506 513 def fill_op rator_combox(self):507 def fill_operator_combox(self): 514 508 """ 515 509 fill the current combobox with the operator … … 527 521 return [self.model1_name, self.model2_name] 528 522 529 def write_string(self, fname, name1, name2):523 def write_string(self, fname, model1_name, model2_name): 530 524 """ 531 525 Write and Save file … … 533 527 self.fname = fname 534 528 description = self.desc_tcl.GetValue().lstrip().rstrip() 535 if description == '': 536 description = name1 + self._operator + name2 537 text = self._operator_choice.GetValue() 538 if text.count('+') > 0: 539 factor = 'scale_factor' 540 f_oper = '*' 541 default_val = '1.0' 542 else: 543 factor = 'BackGround' 544 f_oper = '+' 545 default_val = '0.0' 546 path = self.fname 547 try: 548 out_f = open(path, 'w') 549 except: 550 raise 551 lines = SUM_TEMPLATE.split('\n') 552 for line in lines: 553 try: 554 if line.count("scale_factor"): 555 line = line.replace('scale_factor', factor) 556 #print "scale_factor", line 557 if line.count("= %s"): 558 out_f.write(line % (default_val) + "\n") 559 elif line.count("import Model as P1"): 560 if self.is_p1_custom: 561 line = line.replace('#', '') 562 out_f.write(line % name1 + "\n") 563 else: 564 out_f.write(line + "\n") 565 elif line.count("import %s as P1"): 566 if not self.is_p1_custom: 567 line = line.replace('#', '') 568 out_f.write(line % (name1) + "\n") 569 else: 570 out_f.write(line + "\n") 571 elif line.count("import Model as P2"): 572 if self.is_p2_custom: 573 line = line.replace('#', '') 574 out_f.write(line % name2 + "\n") 575 else: 576 out_f.write(line + "\n") 577 elif line.count("import %s as P2"): 578 if not self.is_p2_custom: 579 line = line.replace('#', '') 580 out_f.write(line % (name2) + "\n") 581 else: 582 out_f.write(line + "\n") 583 elif line.count("P1 = find_model"): 584 out_f.write(line % (name1) + "\n") 585 elif line.count("P2 = find_model"): 586 out_f.write(line % (name2) + "\n") 587 588 elif line.count("self.description = '%s'"): 589 out_f.write(line % description + "\n") 590 #elif line.count("run") and line.count("%s"): 591 # out_f.write(line % self._operator + "\n") 592 #elif line.count("evalDistribution") and line.count("%s"): 593 # out_f.write(line % self._operator + "\n") 594 elif line.count("return") and line.count("%s") == 2: 595 #print "line return", line 596 out_f.write(line % (f_oper, self._operator) + "\n") 597 elif line.count("out2")and line.count("%s"): 598 out_f.write(line % self._operator + "\n") 599 else: 600 out_f.write(line + "\n") 601 except: 602 raise 603 out_f.close() 604 #else: 605 # msg = "Name exists already." 529 desc_line = '' 530 if description.strip() != '': 531 # Sasmodels generates a description for us. If the user provides 532 # their own description, add a line to overwrite the sasmodels one 533 desc_line = "\nmodel_info.description = '{}'".format(description) 534 name = os.path.splitext(os.path.basename(self.fname))[0] 535 output = SUM_TEMPLATE.format(name=name, model1=model1_name, 536 model2=model2_name, operator=self._operator, desc_line=desc_line) 537 with open(self.fname, 'w') as out_f: 538 out_f.write(output) 606 539 607 540 def compile_file(self, path): … … 1278 1211 """ 1279 1212 SUM_TEMPLATE = """ 1280 # A sample of an experimental model function for Sum/Multiply(Pmodel1,Pmodel2) 1281 import os 1282 import sys 1283 import copy 1284 import collections 1285 1286 import numpy 1287 1288 from sas.sascalc.fit.pluginmodel import Model1DPlugin 1289 from sasmodels.sasview_model import find_model 1290 1291 class Model(Model1DPlugin): 1292 name = os.path.splitext(os.path.basename(__file__))[0] 1293 is_multiplicity_model = False 1294 def __init__(self, multiplicity=1): 1295 Model1DPlugin.__init__(self, name='') 1296 P1 = find_model('%s') 1297 P2 = find_model('%s') 1298 p_model1 = P1() 1299 p_model2 = P2() 1300 ## Setting model name model description 1301 self.description = '%s' 1302 if self.name.rstrip().lstrip() == '': 1303 self.name = self._get_name(p_model1.name, p_model2.name) 1304 if self.description.rstrip().lstrip() == '': 1305 self.description = p_model1.name 1306 self.description += p_model2.name 1307 self.fill_description(p_model1, p_model2) 1308 1309 ## Define parameters 1310 self.params = collections.OrderedDict() 1311 1312 ## Parameter details [units, min, max] 1313 self.details = {} 1314 ## Magnetic Panrameters 1315 self.magnetic_params = [] 1316 # non-fittable parameters 1317 self.non_fittable = p_model1.non_fittable 1318 self.non_fittable += p_model2.non_fittable 1319 1320 ##models 1321 self.p_model1= p_model1 1322 self.p_model2= p_model2 1323 1324 1325 ## dispersion 1326 self._set_dispersion() 1327 ## Define parameters 1328 self._set_params() 1329 ## New parameter:scaling_factor 1330 self.params['scale_factor'] = %s 1331 1332 ## Parameter details [units, min, max] 1333 self._set_details() 1334 self.details['scale_factor'] = ['', 0.0, numpy.inf] 1335 1336 1337 #list of parameter that can be fitted 1338 self._set_fixed_params() 1339 1340 ## parameters with orientation 1341 self.orientation_params = [] 1342 for item in self.p_model1.orientation_params: 1343 new_item = "p1_" + item 1344 if not new_item in self.orientation_params: 1345 self.orientation_params.append(new_item) 1346 1347 for item in self.p_model2.orientation_params: 1348 new_item = "p2_" + item 1349 if not new_item in self.orientation_params: 1350 self.orientation_params.append(new_item) 1351 ## magnetic params 1352 self.magnetic_params = [] 1353 for item in self.p_model1.magnetic_params: 1354 new_item = "p1_" + item 1355 if not new_item in self.magnetic_params: 1356 self.magnetic_params.append(new_item) 1357 1358 for item in self.p_model2.magnetic_params: 1359 new_item = "p2_" + item 1360 if not new_item in self.magnetic_params: 1361 self.magnetic_params.append(new_item) 1362 # get multiplicity if model provide it, else 1. 1363 try: 1364 multiplicity1 = p_model1.multiplicity 1365 try: 1366 multiplicity2 = p_model2.multiplicity 1367 except: 1368 multiplicity2 = 1 1369 except: 1370 multiplicity1 = 1 1371 multiplicity2 = 1 1372 ## functional multiplicity of the model 1373 self.multiplicity1 = multiplicity1 1374 self.multiplicity2 = multiplicity2 1375 self.multiplicity_info = [] 1376 1377 def _clone(self, obj): 1378 import copy 1379 obj.params = copy.deepcopy(self.params) 1380 obj.description = copy.deepcopy(self.description) 1381 obj.details = copy.deepcopy(self.details) 1382 obj.dispersion = copy.deepcopy(self.dispersion) 1383 obj.p_model1 = self.p_model1.clone() 1384 obj.p_model2 = self.p_model2.clone() 1385 #obj = copy.deepcopy(self) 1386 return obj 1387 1388 def _get_name(self, name1, name2): 1389 p1_name = self._get_upper_name(name1) 1390 if not p1_name: 1391 p1_name = name1 1392 name = p1_name 1393 name += "_and_" 1394 p2_name = self._get_upper_name(name2) 1395 if not p2_name: 1396 p2_name = name2 1397 name += p2_name 1398 return name 1399 1400 def _get_upper_name(self, name=None): 1401 if name is None: 1402 return "" 1403 upper_name = "" 1404 str_name = str(name) 1405 for index in range(len(str_name)): 1406 if str_name[index].isupper(): 1407 upper_name += str_name[index] 1408 return upper_name 1409 1410 def _set_dispersion(self): 1411 self.dispersion = collections.OrderedDict() 1412 ##set dispersion only from p_model 1413 for name , value in self.p_model1.dispersion.iteritems(): 1414 #if name.lower() not in self.p_model1.orientation_params: 1415 new_name = "p1_" + name 1416 self.dispersion[new_name]= value 1417 for name , value in self.p_model2.dispersion.iteritems(): 1418 #if name.lower() not in self.p_model2.orientation_params: 1419 new_name = "p2_" + name 1420 self.dispersion[new_name]= value 1421 1422 def function(self, x=0.0): 1423 return 0 1424 1425 def getProfile(self): 1426 try: 1427 x,y = self.p_model1.getProfile() 1428 except: 1429 x = None 1430 y = None 1431 1432 return x, y 1433 1434 def _set_params(self): 1435 for name , value in self.p_model1.params.iteritems(): 1436 # No 2D-supported 1437 #if name not in self.p_model1.orientation_params: 1438 new_name = "p1_" + name 1439 self.params[new_name]= value 1440 1441 for name , value in self.p_model2.params.iteritems(): 1442 # No 2D-supported 1443 #if name not in self.p_model2.orientation_params: 1444 new_name = "p2_" + name 1445 self.params[new_name]= value 1446 1447 # Set "scale" as initializing 1448 self._set_scale_factor() 1449 1450 1451 def _set_details(self): 1452 for name ,detail in self.p_model1.details.iteritems(): 1453 new_name = "p1_" + name 1454 #if new_name not in self.orientation_params: 1455 self.details[new_name]= detail 1456 1457 for name ,detail in self.p_model2.details.iteritems(): 1458 new_name = "p2_" + name 1459 #if new_name not in self.orientation_params: 1460 self.details[new_name]= detail 1461 1462 def _set_scale_factor(self): 1463 pass 1464 1465 1466 def setParam(self, name, value): 1467 # set param to this (p1, p2) model 1468 self._setParamHelper(name, value) 1469 1470 ## setParam to p model 1471 model_pre = '' 1472 new_name = '' 1473 name_split = name.split('_', 1) 1474 if len(name_split) == 2: 1475 model_pre = name.split('_', 1)[0] 1476 new_name = name.split('_', 1)[1] 1477 if model_pre == "p1": 1478 if new_name in self.p_model1.getParamList(): 1479 self.p_model1.setParam(new_name, value) 1480 elif model_pre == "p2": 1481 if new_name in self.p_model2.getParamList(): 1482 self.p_model2.setParam(new_name, value) 1483 elif name == 'scale_factor': 1484 self.params['scale_factor'] = value 1485 else: 1486 raise ValueError, "Model does not contain parameter %s" % name 1487 1488 def getParam(self, name): 1489 # Look for dispersion parameters 1490 toks = name.split('.') 1491 if len(toks)==2: 1492 for item in self.dispersion.keys(): 1493 # 2D not supported 1494 if item.lower()==toks[0].lower(): 1495 for par in self.dispersion[item]: 1496 if par.lower() == toks[1].lower(): 1497 return self.dispersion[item][par] 1498 else: 1499 # Look for standard parameter 1500 for item in self.params.keys(): 1501 if item.lower()==name.lower(): 1502 return self.params[item] 1503 return 1504 #raise ValueError, "Model does not contain parameter %s" % name 1505 1506 def _setParamHelper(self, name, value): 1507 # Look for dispersion parameters 1508 toks = name.split('.') 1509 if len(toks)== 2: 1510 for item in self.dispersion.keys(): 1511 if item.lower()== toks[0].lower(): 1512 for par in self.dispersion[item]: 1513 if par.lower() == toks[1].lower(): 1514 self.dispersion[item][par] = value 1515 return 1516 else: 1517 # Look for standard parameter 1518 for item in self.params.keys(): 1519 if item.lower()== name.lower(): 1520 self.params[item] = value 1521 return 1522 1523 raise ValueError, "Model does not contain parameter %s" % name 1524 1525 1526 def _set_fixed_params(self): 1527 self.fixed = [] 1528 for item in self.p_model1.fixed: 1529 new_item = "p1" + item 1530 self.fixed.append(new_item) 1531 for item in self.p_model2.fixed: 1532 new_item = "p2" + item 1533 self.fixed.append(new_item) 1534 1535 self.fixed.sort() 1536 1537 1538 def run(self, x = 0.0): 1539 self._set_scale_factor() 1540 return self.params['scale_factor'] %s \ 1541 (self.p_model1.run(x) %s self.p_model2.run(x)) 1542 1543 def runXY(self, x = 0.0): 1544 self._set_scale_factor() 1545 return self.params['scale_factor'] %s \ 1546 (self.p_model1.runXY(x) %s self.p_model2.runXY(x)) 1547 1548 ## Now (May27,10) directly uses the model eval function 1549 ## instead of the for-loop in Base Component. 1550 def evalDistribution(self, x = []): 1551 self._set_scale_factor() 1552 return self.params['scale_factor'] %s \ 1553 (self.p_model1.evalDistribution(x) %s \ 1554 self.p_model2.evalDistribution(x)) 1555 1556 def set_dispersion(self, parameter, dispersion): 1557 value= None 1558 new_pre = parameter.split("_", 1)[0] 1559 new_parameter = parameter.split("_", 1)[1] 1560 try: 1561 if new_pre == 'p1' and \ 1562 new_parameter in self.p_model1.dispersion.keys(): 1563 value= self.p_model1.set_dispersion(new_parameter, dispersion) 1564 if new_pre == 'p2' and \ 1565 new_parameter in self.p_model2.dispersion.keys(): 1566 value= self.p_model2.set_dispersion(new_parameter, dispersion) 1567 self._set_dispersion() 1568 return value 1569 except: 1570 raise 1571 1572 def fill_description(self, p_model1, p_model2): 1573 description = "" 1574 description += "This model gives the summation or multiplication of" 1575 description += "%s and %s. "% ( p_model1.name, p_model2.name ) 1576 self.description += description 1577 1578 if __name__ == "__main__": 1579 m1= Model() 1580 #m1.setParam("p1_scale", 25) 1581 #m1.setParam("p1_length", 1000) 1582 #m1.setParam("p2_scale", 100) 1583 #m1.setParam("p2_rg", 100) 1584 out1 = m1.runXY(0.01) 1585 1586 m2= Model() 1587 #m2.p_model1.setParam("scale", 25) 1588 #m2.p_model1.setParam("length", 1000) 1589 #m2.p_model2.setParam("scale", 100) 1590 #m2.p_model2.setParam("rg", 100) 1591 out2 = m2.p_model1.runXY(0.01) %s m2.p_model2.runXY(0.01)\n 1592 print "My name is %s."% m1.name 1593 print out1, " = ", out2 1594 if out1 == out2: 1595 print "===> Simple Test: Passed!" 1596 else: 1597 print "===> Simple Test: Failed!" 1213 from sasmodels.core import load_model_info 1214 from sasmodels.sasview_model import make_model_from_info 1215 1216 model_info = load_model_info('{model1}{operator}{model2}') 1217 model_info.name = '{name}'{desc_line} 1218 Model = make_model_from_info(model_info) 1598 1219 """ 1599 1600 1220 if __name__ == "__main__": 1601 1221 # app = wx.PySimpleApp() -
src/sas/sasgui/perspectives/fitting/fitting.py
r489f53a r2d9526d 1742 1742 @param unsmeared_error: data error, rescaled to unsmeared model 1743 1743 """ 1744 1745 1744 number_finite = np.count_nonzero(np.isfinite(y)) 1746 1745 np.nan_to_num(y) … … 1748 1747 data_description=model.name, 1749 1748 data_id=str(page_id) + " " + data.name) 1749 plots_to_update = [] # List of plottables that have changed since last calculation 1750 # Create the new theories 1750 1751 if unsmeared_model is not None: 1751 self.create_theory_1D(x, unsmeared_model, page_id, model, data, state, 1752 unsmeared_model_plot = self.create_theory_1D(x, unsmeared_model, 1753 page_id, model, data, state, 1752 1754 data_description=model.name + " unsmeared", 1753 1755 data_id=str(page_id) + " " + data.name + " unsmeared") 1756 plots_to_update.append(unsmeared_model_plot) 1754 1757 1755 1758 if unsmeared_data is not None and unsmeared_error is not None: 1756 self.create_theory_1D(x, unsmeared_data, page_id, model, data, state, 1759 unsmeared_data_plot = self.create_theory_1D(x, unsmeared_data, 1760 page_id, model, data, state, 1757 1761 data_description="Data unsmeared", 1758 1762 data_id="Data " + data.name + " unsmeared", 1759 1763 dy=unsmeared_error) 1760 # Comment this out until we can get P*S models with correctly populated parameters 1761 #if sq_model is not None and pq_model is not None: 1762 # self.create_theory_1D(x, sq_model, page_id, model, data, state, 1763 # data_description=model.name + " S(q)", 1764 # data_id=str(page_id) + " " + data.name + " S(q)") 1765 # self.create_theory_1D(x, pq_model, page_id, model, data, state, 1766 # data_description=model.name + " P(q)", 1767 # data_id=str(page_id) + " " + data.name + " P(q)") 1764 plots_to_update.append(unsmeared_data_plot) 1765 if sq_model is not None and pq_model is not None: 1766 sq_id = str(page_id) + " " + data.name + " S(q)" 1767 sq_plot = self.create_theory_1D(x, sq_model, page_id, model, data, state, 1768 data_description=model.name + " S(q)", 1769 data_id=sq_id) 1770 plots_to_update.append(sq_plot) 1771 pq_id = str(page_id) + " " + data.name + " P(q)" 1772 pq_plot = self.create_theory_1D(x, pq_model, page_id, model, data, state, 1773 data_description=model.name + " P(q)", 1774 data_id=pq_id) 1775 plots_to_update.append(pq_plot) 1776 # Update the P(Q), S(Q) and unsmeared theory plots if they exist 1777 wx.PostEvent(self.parent, NewPlotEvent(plots=plots_to_update, 1778 action='update')) 1768 1779 1769 1780 current_pg = self.fit_panel.get_page_by_id(page_id) -
src/sas/sasgui/perspectives/fitting/media/fitting_help.rst
r5295cf5 rca383a0 195 195 the :ref:`Advanced_Plugin_Editor` . 196 196 197 **SasView version 4.2** made it possible to specify whether a plugin created with 198 the *New Plugin Model* dialog is actually a form factor P(Q) or a structure factor 199 S(Q). To do this, simply add one or other of the following lines under the *import* 200 statements. 201 202 For a form factor:: 203 204 form_factor = True 205 206 or for a structure factor:: 207 208 structure_factor = True 209 210 If the plugin is a structure factor it is *also* necessary to add two variables to 211 the parameter list:: 212 213 parameters = [ 214 ['radius_effective', '', 1, [0.0, numpy.inf], 'volume', ''], 215 ['volfraction', '', 1, [0.0, 1.0], '', ''], 216 [...], 217 218 and to the declarations of the functions Iq and Iqxy::: 219 220 def Iq(x , radius_effective, volfraction, ...): 221 222 def Iqxy(x, y, radius_effective, volfraction, ...): 223 224 Such a plugin should then be available in the S(Q) drop-down box on a FitPage (once 225 a P(Q) model has been selected). 226 197 227 Sum|Multi(p1,p2) 198 228 ^^^^^^^^^^^^^^^^ … … 206 236 or:: 207 237 208 Plugin Model = scale_factor * model_1 /* model_2+ background238 Plugin Model = scale_factor * (model1 * model2) + background 209 239 210 240 In the *Easy Sum/Multi Editor* give the new model a function name and brief 211 241 description (to appear under the *Details* button on the *FitPage*). Then select 212 242 two existing models, as p1 and p2, and the required operator, '+' or '*' between 213 them. Finally, click the *Apply* button to generate the model and then click *Close*. 214 215 Any changes to a plugin model generated in this way only become effective *after* it is re-selected from the model drop-down menu on the FitPage. 243 them. Finally, click the *Apply* button to generate and test the model and then click *Close*. 244 245 Any changes to a plugin model generated in this way only become effective *after* it is re-selected 246 from the plugin models drop-down menu on the FitPage. If the model is not listed you can force a 247 recompilation of the plugins by selecting *Fitting* > *Plugin Model Operations* > *Load Plugin Models*. 248 249 **SasView version 4.2** introduced a much simplified and more extensible structure for plugin models 250 generated through the Easy Sum/Multi Editor. For example, the code for a combination of a sphere model 251 with a power law model now looks like this:: 252 253 from sasmodels.core import load_model_info 254 from sasmodels.sasview_model import make_model_from_info 255 256 model_info = load_model_info('sphere+power_law') 257 model_info.name = 'MyPluginModel' 258 model_info.description = 'sphere + power_law' 259 Model = make_model_from_info(model_info) 260 261 To change the models or operators contributing to this plugin it is only necessary to edit the string 262 in the brackets after *load_model_info*, though it would also be a good idea to update the model name 263 and description too!!! 264 265 The model specification string can handle multiple models and combinations of operators (+ or *) which 266 are processed according to normal conventions. Thus 'model1+model2*model3' would be valid and would 267 multiply model2 by model3 before adding model1. In this example, parameters in the *FitPage* would be 268 prefixed A (for model2), B (for model3) and C (for model1). Whilst this might appear a little 269 confusing, unless you were creating a plugin model from multiple instances of the same model the parameter 270 assignments ought to be obvious when you load the plugin. 271 272 If you need to include another plugin model in the model specification string, just prefix the name of 273 that model with *custom*. For instance:: 274 275 sphere+custom.MyPluginModel 276 277 To create a P(Q)*\S(Q) model use the @ symbol instead of * like this:: 278 279 sphere@hardsphere 280 281 This streamlined approach to building complex plugin models from existing library models, or models 282 available on the *Model Marketplace*, also permits the creation of P(Q)*\S(Q) plugin models, something 283 that was not possible in earlier versions of SasView. 216 284 217 285 .. _Advanced_Plugin_Editor: … … 484 552 .. ZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZ 485 553 554 .. _Batch_Fit_Mode: 555 486 556 Batch Fit Mode 487 557 -------------- … … 636 706 637 707 Example: radius [2 : 5] , radius [10 : 25] 638 639 .. ZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZ 640 641 .. note:: This help document was last changed by Steve King, 10Oct2016 708 709 .. ZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZ 710 711 Combined Batch Fit Mode 712 ----------------------- 713 714 The purpose of the Combined Batch Fit is to allow running two or more batch 715 fits in sequence without overwriting the output table of results. This may be 716 of interest for example if one is fitting a series of data sets where there is 717 a shape change occurring in the series that requires changing the model part 718 way through the series; for example a sphere to rod transition. Indeed the 719 regular batch mode does not allow for multiple models and requires all the 720 files in the series to be fit with single model and set of parameters. While 721 it is of course possible to just run part of the series as a batch fit using 722 model one followed by running another batch fit on the rest of the series with 723 model two (and/or model three etc), doing so will overwrite the table of 724 outputs from the previous batch fit(s). This may not be desirable if one is 725 interested in comparing the parameters: for example the sphere radius of set 726 one and the cylinder radius of set two. 727 728 Method 729 ^^^^^^ 730 731 In order to use the *Combined Batch Fit*, first load all the data needed as 732 described in :ref:`Loading_data`. Next start up two or more *BatchPage* fits 733 following the instructions in :ref:`Batch_Fit_Mode` but **DO NOT PRESS FIT**. 734 At this point the *Combine Batch Fit* menu item under the *Fitting menu* should 735 be active (if there is one or no *BatchPage* the menu item will be greyed out 736 and inactive). Clicking on *Combine Batch Fit* will bring up a new panel, 737 similar to the *Const & Simult Fit* panel. In this case there will be a 738 checkbox for each *BatchPage* instead of each *FitPage* that should be included 739 in the fit. Once all are selected, click the Fit button on 740 the *BatchPage* to run each batch fit in *sequence* 741 742 .. image:: combine_batch_page.png 743 744 The batch table will then pop up at the end as for the case of the simple Batch 745 Fitting with the following caveats: 746 747 .. note:: 748 The order matters. The parameters in the table will be taken from the model 749 used in the first *BatchPage* of the list. Any parameters from the 750 second and later *BatchPage* s that have the same name as a parameter in the 751 first will show up allowing for plotting of that parameter across the 752 models. The other parameters will not be available in the grid. 753 .. note:: 754 a corralary of the above is that currently models created as a sum|multiply 755 model will not work as desired because the generated model parameters have a 756 p#_ appended to the beginning and thus radius and p1_radius will not be 757 recognized as the same parameter. 758 759 .. image:: combine_batch_grid.png 760 761 In the example shown above the data is a time series with a shifting peak. 762 The first part of the series was fitted using the *broad_peak* model, while 763 the rest of the data were fit using the *gaussian_peak* model. Unfortunately the 764 time is not listed in the file but the file name contains the information. As 765 described in :ref:`Grid_Window`, a column can be added manually, in this case 766 called time, and the peak position plotted against time. 767 768 .. image:: combine_batch_plot.png 769 770 Note the discontinuity in the peak position. This reflects the fact that the 771 Gaussian fit is a rather poor model for the data and is not actually 772 finding the peak. 773 774 .. ZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZ 775 776 .. note:: This help document was last changed by Paul Butler, 10 September 777 2017 -
src/sas/sasgui/perspectives/fitting/media/plugin.rst
r72100ee re081946 18 18 * By writing a model from scratch outside of SasView (only recommended for 19 19 code monkeys!) 20 21 **What follows below is quite technical. If you just want a helping hand to get 22 started creating your own models see** :ref:`Adding_your_own_models`. 20 23 21 24 Overview -
src/sas/sasgui/perspectives/fitting/model_thread.py
r7432acb r0f9ea1c 71 71 (self.data.qy_data * self.data.qy_data)) 72 72 73 # For theory, qmax is based on 1d qmax 73 # For theory, qmax is based on 1d qmax 74 74 # so that must be mulitified by sqrt(2) to get actual max for 2d 75 75 index_model = (self.qmin <= radius) & (radius <= self.qmax) … … 91 91 self.data.qy_data[index_model] 92 92 ]) 93 output = np.zeros(len(self.data.qx_data)) 93 # Initialize output to NaN so masked elements do not get plotted. 94 output = np.empty_like(self.data.qx_data) 94 95 # output default is None 95 96 # This method is to distinguish between masked 96 97 #point(nan) and data point = 0. 97 output = output / output98 output[:] = np.NaN 98 99 # set value for self.mask==True, else still None to Plottools 99 100 output[index_model] = value … … 198 199 output[index] = self.model.evalDistribution(self.data.x[index]) 199 200 201 x=self.data.x[index] 202 y=output[index] 200 203 sq_values = None 201 204 pq_values = None 202 s_model = None203 p_model = None204 205 if isinstance(self.model, MultiplicationModel): 205 206 s_model = self.model.s_model 206 207 p_model = self.model.p_model 207 elif hasattr(self.model, "get_composition_models"): 208 p_model, s_model = self.model.get_composition_models() 209 210 if p_model is not None and s_model is not None: 211 sq_values = np.zeros((len(self.data.x))) 212 pq_values = np.zeros((len(self.data.x))) 213 sq_values[index] = s_model.evalDistribution(self.data.x[index]) 214 pq_values[index] = p_model.evalDistribution(self.data.x[index]) 208 sq_values = s_model.evalDistribution(x) 209 pq_values = p_model.evalDistribution(x) 210 elif hasattr(self.model, "calc_composition_models"): 211 results = self.model.calc_composition_models(x) 212 if results is not None: 213 pq_values, sq_values = results 214 215 215 216 216 elapsed = time.time() - self.starttime 217 217 218 self.complete(x= self.data.x[index], y=output[index],218 self.complete(x=x, y=y, 219 219 page_id=self.page_id, 220 220 state=self.state, -
src/sas/sasgui/perspectives/fitting/simfitpage.py
r959eb01 ra9f9ca4 1 1 """ 2 Simultaneous fit page2 Simultaneous or Batch fit page 3 3 """ 4 # Note that this is used for both Simultaneous/Constrained fit AND for 5 # combined batch fit. This is done through setting of the batch_on parameter. 6 # There are the a half dozen or so places where an if statement is used as in 7 # if not batch_on: 8 # xxxx 9 # else: 10 # xxxx 11 # This is just wrong but dont have time to fix this go. Proper approach would be 12 # to strip all parts of the code that depend on batch_on and create the top 13 # level class from which a contrained/simultaneous fit page and a combined 14 # batch page inherit. 15 # 16 # 04/09/2017 --PDB 17 4 18 import sys 5 19 from collections import namedtuple … … 400 414 # General Help button 401 415 self.btHelp = wx.Button(self, wx.ID_HELP, 'HELP') 402 self.btHelp.SetToolTipString("Simultaneous/Constrained Fitting help.") 416 if self.batch_on: 417 self.btHelp.SetToolTipString("Combined Batch Fitting help.") 418 else: 419 self.btHelp.SetToolTipString("Simultaneous/Constrained Fitting help.") 403 420 self.btHelp.Bind(wx.EVT_BUTTON, self._on_help) 404 421 … … 527 544 """ 528 545 _TreeLocation = "user/sasgui/perspectives/fitting/fitting_help.html" 529 _PageAnchor = "#simultaneous-fit-mode" 530 _doc_viewer = DocumentationWindow(self, self.ID_DOC, _TreeLocation, 546 if not self.batch_on: 547 _PageAnchor = "#simultaneous-fit-mode" 548 _doc_viewer = DocumentationWindow(self, self.ID_DOC, _TreeLocation, 531 549 _PageAnchor, 532 550 "Simultaneous/Constrained Fitting Help") 551 else: 552 _PageAnchor = "#combined-batch-fit-mode" 553 _doc_viewer = DocumentationWindow(self, self.ID_DOC, _TreeLocation, 554 _PageAnchor, 555 "Combined Batch Fit Help") 533 556 534 557 def set_manager(self, manager): -
src/sas/sasgui/plottools/plottables.py
r45dffa69 r2d9526d 239 239 def replace(self, plottable): 240 240 """Replace an existing plottable from the graph""" 241 selected_color = None 241 # If the user has set a custom color, ensure the new plot is the same color 242 selected_color = plottable.custom_color 242 243 selected_plottable = None 243 244 for p in self.plottables.keys(): 244 245 if plottable.id == p.id: 245 246 selected_plottable = p 246 selected_color = self.plottables[p] 247 if selected_color is None: 248 selected_color = self.plottables[p] 247 249 break 248 if 250 if selected_plottable is not None and selected_color is not None: 249 251 del self.plottables[selected_plottable] 252 plottable.custom_color = selected_color 250 253 self.plottables[plottable] = selected_color 251 254 -
src/sas/sasgui/__init__.py
rd7bb526 rc6bdb3b 1 import sys 2 import os 3 from os.path import exists, expanduser, dirname, realpath, join as joinpath 4 5 6 def dirn(path, n): 7 path = realpath(path) 8 for _ in range(n): 9 path = dirname(path) 10 return path 11 12 # Set up config directories 13 def make_user_folder(): 14 path = joinpath(expanduser("~"),'.sasview') 15 if not exists(path): 16 os.mkdir(path) 17 return path 18 19 20 def find_app_folder(): 21 # We are starting out with the following info: 22 # __file__ = .../sas/sasgui/__init__.pyc 23 # Check if the sister path .../sas/sasview exists, and use it as the 24 # app directory. This will only be the case if the app is not frozen. 25 path = joinpath(dirn(__file__, 2), 'sasview') 26 if exists(path): 27 return path 28 29 # If we are running frozen, then root is a parent directory 30 if sys.platform == 'darwin': 31 # Here is the path to the file on the mac: 32 # .../Sasview.app/Contents/Resources/lib/python2.7/site-packages.zip/sas/sasgui/__init__.pyc 33 # We want the path to the Resources directory. 34 path = dirn(__file__, 6) 35 elif os.name == 'nt': 36 # Here is the path to the file on windows: 37 # ../Sasview/library.zip/sas/sasgui/__init__.pyc 38 # We want the path to the Sasview directory. 39 path = dirn(__file__, 4) 40 else: 41 raise RuntimeError("Couldn't find the app directory") 42 return path 43 44 USER_FOLDER = make_user_folder() 45 APP_FOLDER = find_app_folder() 46 47 48 def get_app_dir(): 49 return APP_FOLDER 50 51 def get_user_dir(): 52 return USER_FOLDER 53 54 def get_custom_config_path(): 55 dirname = os.path.join(get_user_dir(), 'config') 56 # If the directory doesn't exist, create it 57 if not os.path.exists(dirname): 58 os.makedirs(dirname) 59 path = os.path.join(dirname, "custom_config.py") 60 return path 61 62 _config_cache = None 63 def get_local_config(): 64 global _config_cache 65 if not _config_cache: 66 _config_cache = _load_config() 67 return _config_cache 68 69 def _load_config(): 70 import os 71 import sys 72 import logging 73 from sasmodels.custom import load_module_from_path 74 75 logger = logging.getLogger(__name__) 76 dirname = get_app_dir() 77 filename = 'local_config.py' 78 path = os.path.join(dirname, filename) 79 try: 80 module = load_module_from_path('sas.sasgui.local_config', path) 81 logger.info("GuiManager loaded %s", path) 82 return module 83 except Exception as exc: 84 logger.critical("Error loading %s: %s", path, exc) 85 sys.exit() -
src/sas/sasgui/guiframe/CategoryInstaller.py
r235f514 r914ba0a 14 14 import logging 15 15 from collections import defaultdict, OrderedDict 16 17 from sas.sasgui import get_user_dir 16 18 17 19 USER_FILE = 'categories.json' … … 69 71 70 72 @staticmethod 71 def _get_home_dir():72 """73 returns the users sasview config dir74 """75 return os.path.join(os.path.expanduser("~"), ".sasview")76 77 @staticmethod78 73 def _regenerate_model_dict(master_category_dict): 79 74 """ … … 85 80 by_model_dict = defaultdict(list) 86 81 model_enabled_dict = defaultdict(bool) 87 82 88 83 for category in master_category_dict: 89 84 for (model, enabled) in master_category_dict[category]: … … 96 91 def _regenerate_master_dict(by_model_dict, model_enabled_dict): 97 92 """ 98 regenerates master_category_dict from by_model_dict 93 regenerates master_category_dict from by_model_dict 99 94 and model_enabled_dict 100 95 returns the master category dictionary … … 112 107 returns the user data file, eg .sasview/categories.json.json 113 108 """ 114 return os.path.join( CategoryInstaller._get_home_dir(), USER_FILE)109 return os.path.join(get_user_dir(), USER_FILE) 115 110 116 111 @staticmethod … … 150 145 model_name, enabled = master_category_dict[cat][ind] 151 146 if model_name not in _model_list: 152 del_name = True 147 del_name = True 153 148 try: 154 149 by_model_dict.pop(model_name) -
src/sas/sasgui/guiframe/aboutbox.py
r959eb01 r914ba0a 24 24 import os.path 25 25 import os 26 try: 27 # Try to find a local config 28 import imp 29 path = os.getcwd() 30 if(os.path.isfile("%s/%s.py" % (path, 'local_config'))) or \ 31 (os.path.isfile("%s/%s.pyc" % (path, 'local_config'))): 32 fObj, path, descr = imp.find_module('local_config', [path]) 33 config = imp.load_module('local_config', fObj, path, descr) 34 else: 35 # Try simply importing local_config 36 import local_config as config 37 except: 38 # Didn't find local config, load the default 39 import config 26 27 from sas.sasgui import get_local_config 28 config = get_local_config() 40 29 41 30 def launchBrowser(url): 42 31 """ 43 32 Launches browser and opens specified url 44 33 45 34 In some cases may require BROWSER environment variable to be set up. 46 35 47 36 :param url: URL to open 48 37 49 38 """ 50 39 import webbrowser … … 55 44 """ 56 45 "About" Dialog 57 46 58 47 Shows product name, current version, authors, and link to the product page. 59 48 Current version is taken from version.py 60 61 """ 62 49 50 """ 51 63 52 def __init__(self, *args, **kwds): 64 53 … … 66 55 kwds["style"] = wx.DEFAULT_DIALOG_STYLE 67 56 wx.Dialog.__init__(self, *args, **kwds) 68 57 69 58 file_dir = os.path.dirname(__file__) 70 59 71 60 # Mac doesn't display images with transparent background so well, 72 61 # keep it for Windows 73 62 image = file_dir + "/images/angles_flat.png" 74 63 75 64 if os.path.isfile(config._corner_image): 76 65 image = config._corner_image … … 80 69 else: 81 70 self.bitmap_logo = wx.StaticBitmap(self, -1, wx.Bitmap(image)) 82 71 83 72 self.label_title = wx.StaticText(self, -1, config.__appname__) 84 73 self.label_version = wx.StaticText(self, -1, "") … … 112 101 #self.bitmap_button_danse = wx.BitmapButton(self, -1, wx.NullBitmap) 113 102 self.bitmap_button_msu = wx.BitmapButton(self, -1, wx.NullBitmap) 114 103 115 104 self.bitmap_button_isis = wx.BitmapButton(self, -1, wx.NullBitmap) 116 105 self.bitmap_button_ess = wx.BitmapButton(self, -1, wx.NullBitmap) … … 119 108 self.bitmap_button_tudelft = wx.BitmapButton(self, -1, wx.NullBitmap) 120 109 self.bitmap_button_dls = wx.BitmapButton(self, -1, wx.NullBitmap) 121 110 122 111 self.static_line_3 = wx.StaticLine(self, -1) 123 112 self.button_OK = wx.Button(self, wx.ID_OK, "OK") … … 125 114 self.__set_properties() 126 115 self.__do_layout() 127 116 128 117 self.Bind(wx.EVT_BUTTON, self.onNistLogo, self.bitmap_button_nist) 129 118 self.Bind(wx.EVT_BUTTON, self.onUmdLogo, self.bitmap_button_umd) … … 145 134 random.shuffle(config._authors) 146 135 strLabel = ", ".join(config._authors) 147 136 148 137 # display version and svn revison numbers 149 138 verwords = config.__version__.split('.') … … 157 146 self.label_version.SetLabel(config.__version__)#(version) 158 147 self.label_svnrevision.SetLabel(build_num) 159 148 160 149 # set bitmaps for logo buttons 161 150 image = file_dir + "/images/nist_logo.png" 162 151 if os.path.isfile(config._nist_logo): 163 152 image = config._nist_logo 164 logo = wx.Bitmap(image) 153 logo = wx.Bitmap(image) 165 154 self.bitmap_button_nist.SetBitmapLabel(logo) 166 155 167 156 image = file_dir + "/images/umd_logo.png" 168 157 if os.path.isfile(config._umd_logo): 169 158 image = config._umd_logo 170 logo = wx.Bitmap(image) 159 logo = wx.Bitmap(image) 171 160 self.bitmap_button_umd.SetBitmapLabel(logo) 172 161 … … 174 163 if os.path.isfile(config._ornl_logo): 175 164 image = config._ornl_logo 176 logo = wx.Bitmap(image) 165 logo = wx.Bitmap(image) 177 166 self.bitmap_button_ornl.SetBitmapLabel(logo) 178 167 … … 181 170 if os.path.isfile(config._sns_logo): 182 171 image = config._sns_logo 183 logo = wx.Bitmap(image) 172 logo = wx.Bitmap(image) 184 173 self.bitmap_button_sns.SetBitmapLabel(logo) 185 174 186 175 image = file_dir + "/images/nsf_logo.png" 187 176 if os.path.isfile(config._nsf_logo): 188 177 image = config._nsf_logo 189 logo = wx.Bitmap(image) 178 logo = wx.Bitmap(image) 190 179 self.bitmap_button_nsf.SetBitmapLabel(logo) 191 180 … … 201 190 logo = wx.Bitmap(image) 202 191 self.bitmap_button_msu.SetBitmapLabel(logo) 203 192 204 193 image = file_dir + "/images/isis_logo.png" 205 194 if os.path.isfile(config._isis_logo): 206 195 image = config._isis_logo 207 logo = wx.Bitmap(image) 196 logo = wx.Bitmap(image) 208 197 self.bitmap_button_isis.SetBitmapLabel(logo) 209 198 … … 213 202 logo = wx.Bitmap(image) 214 203 self.bitmap_button_ess.SetBitmapLabel(logo) 215 204 216 205 image = file_dir + "/images/ill_logo.png" 217 206 if os.path.isfile(config._ill_logo): … … 219 208 logo = wx.Bitmap(image) 220 209 self.bitmap_button_ill.SetBitmapLabel(logo) 221 210 222 211 image = file_dir + "/images/ansto_logo.png" 223 212 if os.path.isfile(config._ansto_logo): … … 225 214 logo = wx.Bitmap(image) 226 215 self.bitmap_button_ansto.SetBitmapLabel(logo) 227 216 228 217 image = file_dir + "/images/tudelft_logo.png" 229 218 if os.path.isfile(config._tudelft_logo): … … 231 220 logo = wx.Bitmap(image) 232 221 self.bitmap_button_tudelft.SetBitmapLabel(logo) 233 222 234 223 image = file_dir + "/images/dls_logo.png" 235 224 if os.path.isfile(config._dls_logo): … … 237 226 logo = wx.Bitmap(image) 238 227 self.bitmap_button_dls.SetBitmapLabel(logo) 239 228 240 229 # resize dialog window to fit version number nicely 241 230 if wx.VERSION >= (2, 7, 2, 0): … … 244 233 size = [self.GetBestFittingSize()[0], self.GetSize()[1]] 245 234 self.Fit() 246 235 247 236 def __set_properties(self): 248 237 """ … … 310 299 sizer_main.Add(self.static_line_2, 0, wx.EXPAND, 0) 311 300 312 sizer_logos.Add(self.bitmap_button_msu, 0, 301 sizer_logos.Add(self.bitmap_button_msu, 0, 313 302 wx.LEFT|wx.ADJUST_MINSIZE, 2) 314 303 #sizer_logos.Add(self.bitmap_button_danse, 0, 315 304 # wx.LEFT|wx.ADJUST_MINSIZE, 2) 316 #sizer_logos.Add(self.bitmap_button_nsf, 0, 305 #sizer_logos.Add(self.bitmap_button_nsf, 0, 317 306 # wx.LEFT|wx.ADJUST_MINSIZE, 2) 318 sizer_logos.Add(self.bitmap_button_umd, 0, 319 wx.LEFT|wx.ADJUST_MINSIZE, 2) 320 sizer_logos.Add(self.bitmap_button_nist, 0, 321 wx.LEFT|wx.ADJUST_MINSIZE, 2) 322 #sizer_logos.Add(self.bitmap_button_sns, 0, 307 sizer_logos.Add(self.bitmap_button_umd, 0, 308 wx.LEFT|wx.ADJUST_MINSIZE, 2) 309 sizer_logos.Add(self.bitmap_button_nist, 0, 310 wx.LEFT|wx.ADJUST_MINSIZE, 2) 311 #sizer_logos.Add(self.bitmap_button_sns, 0, 323 312 # wx.LEFT|wx.ADJUST_MINSIZE, 2) 324 sizer_logos.Add(self.bitmap_button_ornl, 0, 325 wx.LEFT|wx.ADJUST_MINSIZE, 2) 326 sizer_logos.Add(self.bitmap_button_isis, 0, 327 wx.LEFT|wx.ADJUST_MINSIZE, 2) 328 sizer_logos.Add(self.bitmap_button_ess, 0, 329 wx.LEFT|wx.ADJUST_MINSIZE, 2) 330 sizer_logos.Add(self.bitmap_button_ill, 0, 331 wx.LEFT|wx.ADJUST_MINSIZE, 2) 332 sizer_logos.Add(self.bitmap_button_ansto, 0, 333 wx.LEFT|wx.ADJUST_MINSIZE, 2) 334 sizer_logos.Add(self.bitmap_button_tudelft, 0, 335 wx.LEFT|wx.ADJUST_MINSIZE, 2) 336 sizer_logos.Add(self.bitmap_button_dls, 0, 337 wx.LEFT|wx.ADJUST_MINSIZE, 2) 338 313 sizer_logos.Add(self.bitmap_button_ornl, 0, 314 wx.LEFT|wx.ADJUST_MINSIZE, 2) 315 sizer_logos.Add(self.bitmap_button_isis, 0, 316 wx.LEFT|wx.ADJUST_MINSIZE, 2) 317 sizer_logos.Add(self.bitmap_button_ess, 0, 318 wx.LEFT|wx.ADJUST_MINSIZE, 2) 319 sizer_logos.Add(self.bitmap_button_ill, 0, 320 wx.LEFT|wx.ADJUST_MINSIZE, 2) 321 sizer_logos.Add(self.bitmap_button_ansto, 0, 322 wx.LEFT|wx.ADJUST_MINSIZE, 2) 323 sizer_logos.Add(self.bitmap_button_tudelft, 0, 324 wx.LEFT|wx.ADJUST_MINSIZE, 2) 325 sizer_logos.Add(self.bitmap_button_dls, 0, 326 wx.LEFT|wx.ADJUST_MINSIZE, 2) 327 339 328 sizer_logos.Add((10, 50), 0, wx.ADJUST_MINSIZE, 0) 340 329 sizer_main.Add(sizer_logos, 0, wx.EXPAND, 0) 341 330 sizer_main.Add(self.static_line_3, 0, wx.EXPAND, 0) 342 331 sizer_button.Add((20, 40), 1, wx.EXPAND|wx.ADJUST_MINSIZE, 0) 343 sizer_button.Add(self.button_OK, 0, 332 sizer_button.Add(self.button_OK, 0, 344 333 wx.RIGHT|wx.ADJUST_MINSIZE|wx.CENTER, 10) 345 334 sizer_main.Add(sizer_button, 0, wx.EXPAND, 0) … … 350 339 # end wxGlade 351 340 352 def onNistLogo(self, event): 341 def onNistLogo(self, event): 353 342 """ 354 343 """ … … 356 345 launchBrowser(config._nist_url) 357 346 event.Skip() 358 359 def onUmdLogo(self, event): 347 348 def onUmdLogo(self, event): 360 349 """ 361 350 """ … … 363 352 launchBrowser(config._umd_url) 364 353 event.Skip() 365 366 def onOrnlLogo(self, event): 354 355 def onOrnlLogo(self, event): 367 356 """ 368 357 """ … … 370 359 launchBrowser(config._ornl_url) 371 360 event.Skip() 372 373 def onSnsLogo(self, event): 361 362 def onSnsLogo(self, event): 374 363 """ 375 364 """ … … 377 366 launchBrowser(config._sns_url) 378 367 event.Skip() 379 380 def onNsfLogo(self, event): 368 369 def onNsfLogo(self, event): 381 370 """ 382 371 """ … … 394 383 def onUTLogo(self, event): 395 384 """ 396 """ 385 """ 397 386 # wxGlade: DialogAbout.<event_handler> 398 387 launchBrowser(config._inst_url) 399 388 event.Skip() 400 389 401 def onIsisLogo(self, event): 390 def onIsisLogo(self, event): 402 391 """ 403 392 """ … … 415 404 def onIllLogo(self, event): 416 405 """ 417 """ 406 """ 418 407 # wxGlade: DialogAbout.<event_handler> 419 408 launchBrowser(config._ill_url) … … 422 411 def onAnstoLogo(self, event): 423 412 """ 424 """ 413 """ 425 414 # wxGlade: DialogAbout.<event_handler> 426 415 launchBrowser(config._ansto_url) … … 429 418 def onTudelftLogo(self, event): 430 419 """ 431 """ 420 """ 432 421 # wxGlade: DialogAbout.<event_handler> 433 422 launchBrowser(config._tudelft_url) … … 436 425 def onDlsLogo(self, event): 437 426 """ 438 """ 427 """ 439 428 # wxGlade: DialogAbout.<event_handler> 440 429 launchBrowser(config._dls_url) … … 462 451 app = MyApp(0) 463 452 app.MainLoop() 464 465 ##### end of testing code ##################################################### 453 454 ##### end of testing code ##################################################### -
src/sas/sasgui/guiframe/acknowledgebox.py
r74c8cd0 r914ba0a 12 12 import wx.lib.hyperlink 13 13 from wx.lib.expando import ExpandoTextCtrl 14 import random15 import os.path16 import os17 try:18 # Try to find a local config19 import imp20 path = os.getcwd()21 if(os.path.isfile("%s/%s.py" % (path, 'local_config'))) or \22 (os.path.isfile("%s/%s.pyc" % (path, 'local_config'))):23 fObj, path, descr = imp.find_module('local_config', [path])24 config = imp.load_module('local_config', fObj, path, descr)25 else:26 # Try simply importing local_config27 import local_config as config28 except:29 # Didn't find local config, load the default30 import config31 14 15 from sas.sasgui import get_local_config 16 config = get_local_config() 32 17 33 18 class DialogAcknowledge(wx.Dialog): -
src/sas/sasgui/guiframe/config.py
ra1b8fee rc6bdb3b 1 1 """ 2 2 Application settings 3 3 """ 4 4 from __future__ import print_function … … 6 6 import time 7 7 import os 8 import logging 9 8 10 from sas.sasgui.guiframe.gui_style import GUIFRAME 9 11 import sas.sasview 10 import logging11 12 12 13 13 logger = logging.getLogger(__name__) … … 75 75 _ansto_logo = os.path.join(icon_path, "ansto_logo.png") 76 76 _tudelft_logo = os.path.join(icon_path, "tudelft_logo.png") 77 _dls_logo = os.path.join(icon_path, "dls_logo.png") 77 78 _nsf_logo = os.path.join(icon_path, "nsf_logo.png") 78 79 _danse_logo = os.path.join(icon_path, "danse_logo.png") … … 147 148 SAS_OPENCL = None 148 149 150 # Time out for updating sasview 151 UPDATE_TIMEOUT = 2 152 149 153 def printEVT(message): 150 154 if __EVT_DEBUG__: -
src/sas/sasgui/guiframe/customdir.py
r959eb01 rc6bdb3b 1 1 # Setup and find Custom config dir 2 from __future__ import print_function 3 2 4 import os.path 5 import logging 3 6 import shutil 4 7 5 CONF_DIR = 'config' 6 APPLICATION_NAME = 'sasview' 8 from sasmodels.custom import load_module_from_path 7 9 8 def _find_usersasview_dir(): 9 """ 10 Find and return user/.sasview dir 11 """ 12 return os.path.join(os.path.expanduser("~"), ("." + APPLICATION_NAME)) 10 from sas.sasgui import get_custom_config_path, get_app_dir 13 11 14 def _find_customconf_dir(): 15 """ 16 Find path of the config directory. 17 The plugin directory is located in the user's home directory. 18 """ 19 u_dir = _find_usersasview_dir() 20 return os.path.join(u_dir, CONF_DIR) 12 logger = logging.getLogger(__name__) 21 13 22 def _setup_conf_dir(path): 14 _config_cache = None 15 def setup_custom_config(): 23 16 """ 24 17 Setup the custom config dir and cat file 25 18 """ 26 conf_dir = _find_customconf_dir() 27 # If the plugin directory doesn't exist, create it 28 if not os.path.isdir(conf_dir): 29 os.makedirs(conf_dir) 30 config_file = os.path.join(conf_dir, "custom_config.py") 31 32 # Place example user models as needed 33 try: 34 if not os.path.isfile(config_file): 35 shutil.copyfile(os.path.join(path, "custom_config.py"), config_file) 36 37 #Adding SAS_OPENCL if it doesn't exist in the config file 38 # - to support backcompability 39 if not "SAS_OPENCL" in open(config_file).read(): 40 open(config_file,"a+").write("SAS_OPENCL = \"None\"\n") 41 except: 42 # Check for data path next to exe/zip file. 43 #Look for maximum n_dir up of the current dir to find plugins dir 44 n_dir = 12 45 is_dir = False 46 f_dir = path 47 for i in range(n_dir): 48 if i > 1: 49 f_dir, _ = os.path.split(f_dir) 50 temp_path = os.path.join(f_dir, "custom_config.py") 51 if os.path.isfile(temp_path): 52 shutil.copyfile(temp_path, config_file) 53 is_dir = True 54 break 55 if not is_dir: 56 raise 57 return conf_dir 19 global _config_cache 20 if not _config_cache: 21 _config_cache = _setup_custom_config() 22 return _config_cache 58 23 59 24 60 class SetupCustom(object): 61 """ 62 implement custom config dir 63 """ 64 def find_dir(self): 65 return _find_customconf_dir() 66 67 def setup_dir(self, path): 68 return _setup_conf_dir(path) 25 def _setup_custom_config(): 26 path = get_custom_config_path() 27 if not os.path.isfile(path): 28 try: 29 # if the custom config file does not exist, copy the default from 30 # the app dir 31 shutil.copyfile(os.path.join(get_app_dir(), "custom_config.py"), 32 path) 33 except Exception: 34 logger.error("Could not copy default custom config.") 35 36 #Adding SAS_OPENCL if it doesn't exist in the config file 37 # - to support backcompability 38 if not "SAS_OPENCL" in open(path).read(): 39 try: 40 open(config_file, "a+").write("SAS_OPENCL = \"None\"\n") 41 except Exception: 42 logger.error("Could not update custom config with SAS_OPENCL.") 43 44 custom_config = _load_config(path) 45 return custom_config 46 47 48 def _load_config(path): 49 if os.path.exists(path): 50 try: 51 module = load_module_from_path('sas.sasview.custom_config', path) 52 logger.info("GuiManager loaded %s", path) 53 return module 54 except Exception as exc: 55 logger.error("Error loading %s: %s", path, exc) 56 57 from sas.sasview import custom_config 58 logger.info("GuiManager custom_config defaults to sas.sasview.custom_config") 59 return custom_config -
src/sas/sasgui/guiframe/data_panel.py
ra1b8fee rc6bdb3b 33 33 from sas.sasgui.guiframe.local_perspectives.plotting.SimplePlot \ 34 34 import PlotFrame as QucikPlotDialog 35 import sas.sasgui.guiframe.config as config 35 from sas.sasgui import get_local_config 36 37 config = get_local_config() 36 38 37 39 # Check version -
src/sas/sasgui/guiframe/documentation_window.py
r959eb01 rd66dbcc 16 16 import os 17 17 import logging 18 import wx19 18 import webbrowser 20 19 import urllib 21 20 import sys 22 21 22 import wx 23 try: 24 import wx.html2 as html 25 WX_SUPPORTS_HTML2 = True 26 except: 27 WX_SUPPORTS_HTML2 = False 28 29 from sas.sasgui import get_app_dir 30 23 31 logger = logging.getLogger(__name__) 24 32 25 33 SPHINX_DOC_ENV = "SASVIEW_DOC_PATH" 26 WX_SUPPORTS_HTML2 = True27 try:28 import wx.html2 as html29 except:30 WX_SUPPORTS_HTML2 = False31 32 33 from gui_manager import get_app_dir34 35 34 36 35 class DocumentationWindow(wx.Frame): -
src/sas/sasgui/guiframe/gui_manager.py
r2f22db9 r914ba0a 22 22 import re 23 23 import logging 24 import httplib25 24 import traceback 26 25 import urllib 27 import urllib228 26 import json 29 27 28 from matplotlib import _pylab_helpers 29 30 from sas.sasgui import get_local_config, get_app_dir, get_user_dir 30 31 from sas.sasgui.guiframe.events import EVT_CATEGORY 31 32 from sas.sasgui.guiframe.events import EVT_STATUS … … 46 47 from sas.sascalc.dataloader.loader import Loader 47 48 from sas.sasgui.guiframe.proxy import Connection 48 from matplotlib import _pylab_helpers49 from sas.sasgui.guiframe.customdir import setup_custom_config 49 50 50 51 logger = logging.getLogger(__name__) 51 52 52 warnings.simplefilter("ignore") 53 53 54 def get_app_dir(): 55 """ 56 The application directory is the one where the default custom_config.py 57 file resides. 58 59 :returns: app_path - the path to the applicatin directory 60 """ 61 # First, try the directory of the executable we are running 62 app_path = sys.path[0] 63 if os.path.isfile(app_path): 64 app_path = os.path.dirname(app_path) 65 if os.path.isfile(os.path.join(app_path, "custom_config.py")): 66 app_path = os.path.abspath(app_path) 67 logger.info("Using application path: %s", app_path) 68 return app_path 69 70 # Next, try the current working directory 71 if os.path.isfile(os.path.join(os.getcwd(), "custom_config.py")): 72 logger.info("Using application path: %s", os.getcwd()) 73 return os.path.abspath(os.getcwd()) 74 75 # Finally, try the directory of the sasview module 76 # TODO: gui_manager will have to know about sasview until we 77 # clean all these module variables and put them into a config class 78 # that can be passed by sasview.py. 79 logger.debug(sys.executable) 80 logger.debug(str(sys.argv)) 81 from sas import sasview as sasview 82 app_path = os.path.dirname(sasview.__file__) 83 logger.debug("Using application path: %s", app_path) 84 return app_path 85 86 87 def get_user_directory(): 88 """ 89 Returns the user's home directory 90 """ 91 userdir = os.path.join(os.path.expanduser("~"), ".sasview") 92 if not os.path.isdir(userdir): 93 os.makedirs(userdir) 94 return userdir 95 96 97 def _find_local_config(file, path): 98 """ 99 Find configuration file for the current application 100 """ 101 config_module = None 102 fObj = None 103 try: 104 fObj, path_config, descr = imp.find_module(file, [path]) 105 config_module = imp.load_module(file, fObj, path_config, descr) 106 except: 107 logger.error("Error loading %s/%s: %s" % (path, file, sys.exc_value)) 108 finally: 109 if fObj is not None: 110 fObj.close() 111 logger.debug("GuiManager loaded %s/%s" % (path, file)) 112 return config_module 113 114 # Get APP folder 115 PATH_APP = get_app_dir() 116 DATAPATH = PATH_APP 117 118 # GUI always starts from the App folder 119 # os.chdir(PATH_APP) 120 # Read in the local config, which can either be with the main 121 # application or in the installation directory 122 config = _find_local_config('local_config', PATH_APP) 123 if config is None: 124 config = _find_local_config('local_config', os.getcwd()) 125 if config is None: 126 # Didn't find local config, load the default 127 import sas.sasgui.guiframe.config as config 128 logger.debug("using default local_config") 129 else: 130 logger.debug("found local_config in %s" % os.getcwd()) 131 else: 132 logger.debug("found local_config in %s" % PATH_APP) 133 134 from sas.sasgui.guiframe.customdir import SetupCustom 135 c_conf_dir = SetupCustom().setup_dir(PATH_APP) 136 custom_config = _find_local_config('custom_config', c_conf_dir) 137 if custom_config is None: 138 custom_config = _find_local_config('custom_config', os.getcwd()) 139 if custom_config is None: 140 msgConfig = "Custom_config file was not imported" 141 logger.debug(msgConfig) 142 else: 143 logger.debug("using custom_config in %s" % os.getcwd()) 144 else: 145 logger.debug("using custom_config from %s" % c_conf_dir) 54 config = get_local_config() 55 custom_config = setup_custom_config() 146 56 147 57 # read some constants from config … … 177 87 DEFAULT_OPEN_FOLDER = os.path.abspath(open_folder) 178 88 else: 179 DEFAULT_OPEN_FOLDER = PATH_APP89 DEFAULT_OPEN_FOLDER = get_app_dir() 180 90 SAS_OPENCL = custom_config.SAS_OPENCL 181 91 except: … … 192 102 DEFAULT_PERSPECTIVE = None 193 103 CLEANUP_PLOT = False 104 DEFAULT_OPEN_FOLDER = get_app_dir() 194 105 DEFAULT_OPEN_FOLDER = PATH_APP 195 106 SAS_OPENCL = None … … 265 176 if os.path.isfile(ico_file): 266 177 self.SetIcon(wx.Icon(ico_file, wx.BITMAP_TYPE_ICO)) 267 self.path = PATH_APP178 self.path = get_app_dir() 268 179 self.application_name = APPLICATION_NAME 269 180 # Application manager … … 540 451 try: 541 452 fd = open(file_name, 'w') 542 except :453 except Exception: 543 454 # On Permission denied: IOError 544 temp_dir = get_user_dir ectory()455 temp_dir = get_user_dir() 545 456 temp_file_name = os.path.join(temp_dir, name) 546 457 fd = open(temp_file_name, 'w') … … 1532 1443 # want Analysis. This is NOT an issue on the Mac which does not 1533 1444 # have the extra Window menu item. 1534 # March 2016 Code Camp -- PDB 1445 # March 2016 Code Camp -- PDB 1535 1446 Tools_pos = self._menubar.FindMenu("Tools") 1536 1447 self._menubar.Insert(Tools_pos+1, self._applications_menu, … … 2163 2074 logger.info("Failed to connect to www.sasview.org") 2164 2075 self._process_version(version_info, standalone=event is None) 2076 2165 2077 2166 2078 def _process_version(self, version_info, standalone=True): … … 3351 3263 if basename.lower() in [app_py, app_exe, app_app, app_base]: 3352 3264 data_base = sys.argv[1] 3353 input_file = os.path.normpath(os.path.join( DATAPATH,3265 input_file = os.path.normpath(os.path.join(get_app_dir(), 3354 3266 data_base)) 3355 3267 if input_file is None: … … 3366 3278 # do it only the first time app loaded 3367 3279 # delete unused model folder 3368 model_folder = os.path.join( PATH_APP, path)3280 model_folder = os.path.join(get_app_dir(), path) 3369 3281 if os.path.exists(model_folder) and os.path.isdir(model_folder): 3370 3282 if len(os.listdir(model_folder)) > 0: -
src/sas/sasgui/guiframe/local_perspectives/data_loader/data_loader.py
rdcb91cf r759a8ab 12 12 from sas.sascalc.dataloader.loader import Loader 13 13 from sas.sascalc.dataloader.loader_exceptions import NoKnownLoaderException 14 15 from sas.sasgui import get_local_config 14 16 from sas.sasgui.guiframe.plugin_base import PluginBase 15 17 from sas.sasgui.guiframe.events import StatusEvent 16 18 from sas.sasgui.guiframe.gui_style import GUIFRAME 17 19 from sas.sasgui.guiframe.gui_manager import DEFAULT_OPEN_FOLDER 18 try: 19 # Try to find a local config 20 import imp 21 path = os.getcwd() 22 if(os.path.isfile("%s/%s.py" % (path, 'local_config'))) or \ 23 (os.path.isfile("%s/%s.pyc" % (path, 'local_config'))): 24 fObj, path, descr = imp.find_module('local_config', [path]) 25 config = imp.load_module('local_config', fObj, path, descr) 26 else: 27 # Try simply importing local_config 28 import local_config as config 29 except: 30 # Didn't find local config, load the default 31 import sas.sasgui.guiframe.config as config 32 33 if config is None: 34 import sas.sasgui.guiframe.config as config 35 36 20 21 config = get_local_config() 37 22 extension_list = [] 38 23 if config.APPLICATION_STATE_EXTENSION is not None: -
src/sas/sasgui/guiframe/startup_configuration.py
r7432acb r914ba0a 1 2 1 ################################################################################ 3 2 #This software was developed by the University of Tennessee as part of the 4 3 #Distributed Data Analysis of Neutron Scattering Experiments (DANSE) 5 #project funded by the US National Science Foundation. 4 #project funded by the US National Science Foundation. 6 5 # 7 6 #See the license text in license.txt … … 9 8 #copyright 2009, University of Tennessee 10 9 ################################################################################ 10 import os 11 import copy 12 11 13 import wx 12 import os 13 import sys 14 import copy 15 #import sas.sasgui.guiframe.gui_manager as gui 16 from sas.sasgui.guiframe.events import StatusEvent 14 15 from sas.sasgui import get_custom_config_path 16 from sas.sasgui.guiframe.events import StatusEvent 17 17 from sas.sasgui.guiframe.gui_style import GUIFRAME 18 18 from sas.sasgui.guiframe import gui_manager as CURRENT 19 from sas.sasgui.guiframe.customdir import SetupCustom 19 20 20 21 # default configuration 21 22 DEFAULT_STRINGS = {'GUIFRAME_WIDTH':-1, … … 62 63 """ 63 64 def __init__(self, parent, gui, id=-1, title="Startup Setting"): 64 wx.Dialog.__init__(self, parent, id, title, 65 wx.Dialog.__init__(self, parent, id, title, 65 66 size=(PANEL_WIDTH, PANEL_HEIGHT)) 66 67 # parent 67 68 self.parent = parent 68 self.path = SetupCustom().find_dir()69 69 self._gui = gui 70 # font size 70 # font size 71 71 self.SetWindowVariant(variant=FONT_VARIANT) 72 72 self.current_string = copy.deepcopy(CURRENT_STRINGS) … … 76 76 title_text = wx.StaticText(self, id=wx.NewId(), label='Set interface configuration') 77 77 78 default_bt = wx.RadioButton(self, -1, 'Default View', (15, 30), 78 default_bt = wx.RadioButton(self, -1, 'Default View', (15, 30), 79 79 style=wx.RB_GROUP) 80 80 default_bt.Bind(wx.EVT_RADIOBUTTON, self.OnDefault) … … 87 87 note_txt = wx.StaticText(self, -1, msg, (15, 75)) 88 88 note_txt.SetForegroundColour("black") 89 89 90 90 hbox = wx.BoxSizer(wx.HORIZONTAL) 91 91 okButton = wx.Button(self, wx.ID_OK, 'Set', size=(70, 25)) 92 closeButton = wx.Button(self, wx.ID_CANCEL, 'Cancel', size=(70, 25))92 closeButton = wx.Button(self, wx.ID_CANCEL, 'Cancel', size=(70, 25)) 93 93 hbox.Add(closeButton, 1, wx.RIGHT, 5) 94 94 hbox.Add(okButton, 1, wx.RIGHT, 5) … … 102 102 self.SetSizer(vbox) 103 103 104 104 105 105 def OnDefault(self, event=None): 106 106 """ … … 111 111 self.return_string = copy.deepcopy(DEFAULT_STRINGS) 112 112 return self.return_string 113 113 114 114 def OnCurrent(self, event=None): 115 115 """ … … 134 134 p_size = CURRENT_STRINGS['PLOPANEL_WIDTH'] 135 135 self.current_string['PLOPANEL_WIDTH'] = p_size 136 136 137 137 try: 138 138 control_frame = self.parent.get_current_perspective().frame … … 143 143 self.current_string['CONTROL_WIDTH'] = -1 144 144 self.current_string['CONTROL_HEIGHT'] = -1 145 145 146 146 data_pw, _ = self.parent.panels["data_panel"].frame.GetSizeTuple() 147 147 if data_pw is None: 148 148 data_pw = CURRENT_STRINGS['DATAPANEL_WIDTH'] 149 149 self.current_string['DATAPANEL_WIDTH'] = data_pw 150 150 151 151 #label = self.parent._data_panel_menu.GetText() 152 152 label = self.parent.panels['data_panel'].frame.IsShown() … … 155 155 else: 156 156 self.current_string['DATALOADER_SHOW'] = False 157 157 158 158 if self.parent._toolbar.IsShown(): 159 159 self.current_string['TOOLBAR_SHOW'] = True 160 160 else: 161 161 self.current_string['TOOLBAR_SHOW'] = False 162 162 163 163 style = self._gui & GUIFRAME.FLOATING_PANEL 164 if style == GUIFRAME.FLOATING_PANEL: 164 if style == GUIFRAME.FLOATING_PANEL: 165 165 self.current_string['FIXED_PANEL'] = False 166 166 else: 167 167 self.current_string['FIXED_PANEL'] = True 168 168 169 169 if self.parent.panels['default'].frame.IsShown(): 170 170 self.current_string['WELCOME_PANEL_SHOW'] = True … … 182 182 self.current_string['DEFAULT_OPEN_FOLDER'] = location 183 183 #self.parent._default_save_location.ascii_letters 184 184 185 185 except: 186 186 raise … … 188 188 self.return_string = self.current_string 189 189 return self.return_string 190 190 191 191 192 def write_custom_config(self): 192 193 """ 193 Write custom configuration 194 """ 195 fname = os.path.join(self.path, 'custom_config.py') 196 self.write_string(fname, self.return_string) 197 198 def write_string(self, fname, strings): 199 """ 200 Write and Save file 201 """ 202 203 try: 204 out_f = open(fname,'w') 205 except : 206 raise #RuntimeError, "Error: Can not change the configuration..." 207 out_f.write("#Application appearance custom configuration\n" ) 208 for key, item in strings.iteritems(): 209 if (key == 'DEFAULT_PERSPECTIVE') or \ 210 (key == 'DEFAULT_OPEN_FOLDER' and item is not None): 211 out_f.write("%s = \"%s\"\n" % (key,str(item))) 212 else: 213 out_f.write("%s = %s\n" % (key,str(item))) 214 215 out_f.close() 216 194 Write custom configuration 195 """ 196 path = get_custom_config_path() 197 with open(path, 'w') as out_f: 198 out_f.write("#Application appearance custom configuration\n") 199 for key, item in self.return_string.iteritems(): 200 if (key == 'DEFAULT_PERSPECTIVE') or \ 201 (key == 'DEFAULT_OPEN_FOLDER' and item != None): 202 out_f.write("%s = \"%s\"\n" % (key, str(item))) 203 else: 204 out_f.write("%s = %s\n" % (key, str(item))) -
src/sas/sasgui/perspectives/fitting/models.py
rb1c2011 r12f7f24 14 14 import py_compile 15 15 import shutil 16 from sasmodels.sasview_model import load_custom_model, load_standard_models 16 17 # Explicitly import from the pluginmodel module so that py2exe 17 18 # places it in the distribution. The Model1DPlugin class is used 18 19 # as the base class of plug-in models. 20 from sas.sasgui import get_user_dir 19 21 from sas.sascalc.fit.pluginmodel import Model1DPlugin 20 22 from sas.sasgui.guiframe.CategoryInstaller import CategoryInstaller 21 from sasmodels.sasview_model import load_custom_model, load_standard_models22 23 23 24 logger = logging.getLogger(__name__) … … 25 26 26 27 PLUGIN_DIR = 'plugin_models' 27 PLUGIN_LOG = os.path.join(os.path.expanduser("~"), '.sasview', PLUGIN_DIR, 28 "plugins.log") 28 PLUGIN_LOG = os.path.join(get_user_dir(), PLUGIN_DIR, "plugins.log") 29 29 PLUGIN_NAME_BASE = '[plug-in] ' 30 30 -
src/sas/sasgui/plottools/__init__.py
rd7bb526 refe730d 1 import config2 1 from PlotPanel import PlotPanel 3 2 from plottables import Data1D, Theory1D -
src/sas/sasgui/plottools/config.py
rd7bb526 ra2a1c20 38 38 import pkg_resources 39 39 pkg_resources.require("matplotlib>=" + plot_version) 40 except :40 except ImportError: 41 41 from distutils.version import LooseVersion as Version 42 42 if Version(matplotlib.__version__) < Version(plot_version):
Note: See TracChangeset
for help on using the changeset viewer.