Changeset 5062bbf in sasview for sansview/perspectives/fitting
- Timestamp:
- Jun 7, 2010 10:26:43 AM (15 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, costrafo411, magnetic_scatt, release-4.1.1, release-4.1.2, release-4.2.2, release_4.0.1, ticket-1009, ticket-1094-headless, ticket-1242-2d-resolution, ticket-1243, ticket-1249, ticket885, unittest-saveload
- Children:
- b94945d
- Parents:
- 79ac6f8
- Location:
- sansview/perspectives/fitting
- Files:
-
- 1 added
- 13 edited
Legend:
- Unmodified
- Added
- Removed
-
sansview/perspectives/fitting/basepage.py
rbb70474 r5062bbf 31 31 class BasicPage(wx.ScrolledWindow): 32 32 """ 33 33 This class provide general structure of fitpanel page 34 34 """ 35 35 ## Internal name for the AUI manager … … 39 39 40 40 def __init__(self,parent, page_info): 41 """ 42 """ 41 43 wx.ScrolledWindow.__init__(self, parent, 42 44 style= wx.FULL_REPAINT_ON_RESIZE ) … … 149 151 class ModelTextCtrl(wx.TextCtrl): 150 152 """ 151 Text control for model and fit parameters. 152 Binds the appropriate events for user interactions. 153 Default callback methods can be overwritten on initialization 154 155 @param kill_focus_callback: callback method for EVT_KILL_FOCUS event 156 @param set_focus_callback: callback method for EVT_SET_FOCUS event 157 @param mouse_up_callback: callback method for EVT_LEFT_UP event 158 @param text_enter_callback: callback method for EVT_TEXT_ENTER event 153 Text control for model and fit parameters. 154 Binds the appropriate events for user interactions. 155 Default callback methods can be overwritten on initialization 156 157 :param kill_focus_callback: callback method for EVT_KILL_FOCUS event 158 :param set_focus_callback: callback method for EVT_SET_FOCUS event 159 :param mouse_up_callback: callback method for EVT_LEFT_UP event 160 :param text_enter_callback: callback method for EVT_TEXT_ENTER event 161 159 162 """ 160 163 ## Set to True when the mouse is clicked while the whole string is selected … … 184 187 def _on_set_focus(self, event): 185 188 """ 186 Catch when the text control is set in focus to highlight the whole 187 text if necessary 188 @param event: mouse event 189 Catch when the text control is set in focus to highlight the whole 190 text if necessary 191 192 :param event: mouse event 193 189 194 """ 190 195 … … 197 202 def _highlight_text(self, event): 198 203 """ 199 Highlight text of a TextCtrl only of no text has be selected 200 @param event: mouse event 204 Highlight text of a TextCtrl only of no text has be selected 205 206 :param event: mouse event 207 201 208 """ 202 209 # Make sure the mouse event is available to other listeners … … 215 222 def _silent_kill_focus(self,event): 216 223 """ 217 224 do nothing to kill focus 218 225 """ 219 226 event.Skip() … … 222 229 def set_page_info(self, page_info): 223 230 """ 224 231 set some page important information at once 225 232 """ 226 233 ##window_name … … 243 250 def onContextMenu(self, event): 244 251 """ 245 252 Retrieve the state selected state 246 253 """ 247 254 # Skipping the save state functionality for release 0.9.0 … … 256 263 def onUndo(self, event): 257 264 """ 258 265 Cancel the previous action 259 266 """ 260 267 #print "enable undo" … … 262 269 wx.PostEvent(self.parent, event) 263 270 264 265 271 def onRedo(self, event): 266 272 """ 267 273 Restore the previous action cancelled 268 274 """ 269 275 #print "enable redo" 270 276 event = NextStateEvent(page= self) 271 277 wx.PostEvent(self.parent, event) 272 273 278 274 279 def define_page_structure(self): 275 280 """ 276 281 Create empty sizer for a panel 277 282 """ 278 283 self.vbox = wx.BoxSizer(wx.VERTICAL) … … 303 308 def set_layout(self): 304 309 """ 305 310 layout 306 311 """ 307 312 self.vbox.Layout() … … 311 316 self.set_scroll() 312 317 self.Centre() 313 314 318 315 319 def set_scroll(self): 320 """ 321 """ 316 322 self.SetScrollbars(20,20,25,65) 317 323 self.Layout() 318 324 self.SetAutoLayout(True) 319 325 320 321 326 def set_owner(self,owner): 322 327 """ 323 set owner of fitpage 324 @param owner: the class responsible of plotting 328 set owner of fitpage 329 330 :param owner: the class responsible of plotting 331 325 332 """ 326 333 self.event_owner = owner … … 329 336 def get_data(self): 330 337 """ 331 338 return the current data 332 339 """ 333 340 return self.data … … 335 342 def set_manager(self, manager): 336 343 """ 337 set panel manager 338 @param manager: instance of plugin fitting 344 set panel manager 345 346 :param manager: instance of plugin fitting 347 339 348 """ 340 349 self.manager = manager … … 343 352 def populate_box(self, dict): 344 353 """ 345 Store list of model 346 @param dict: dictionary containing list of models 354 Store list of model 355 356 :param dict: dictionary containing list of models 357 347 358 """ 348 359 self.model_list_box = dict … … 351 362 def initialize_combox(self): 352 363 """ 353 364 put default value in the combobox 354 365 """ 355 366 ## fill combox box … … 377 388 def set_dispers_sizer(self): 378 389 """ 379 390 fill sizer containing dispersity info 380 391 """ 381 392 self.sizer4.Clear(True) … … 439 450 self.state.enable_disp= self.enable_disp.GetValue() 440 451 self.state.disable_disp= self.disable_disp.GetValue() 441 442 452 443 453 def select_disp_angle(self, event): 444 454 """ 445 Event for when a user select a parameter to average over. 446 @param event: radiobutton event 455 Event for when a user select a parameter to average over. 456 457 :param event: radiobutton event 458 447 459 """ 448 460 self.values=[] … … 533 545 def onResetModel(self, event): 534 546 """ 535 547 Reset model state 536 548 """ 537 549 ## post help message for the selected model … … 550 562 def on_save_state(self, event): 551 563 """ 552 564 Save the current state into file 553 565 """ 554 566 self.save_current_state() … … 569 581 def on_bookmark(self, event): 570 582 """ 571 583 save history of the data and model 572 584 """ 573 585 if self.model==None: … … 616 628 def onSetFocus(self, evt): 617 629 """ 618 highlight the current textcrtl and hide the error text control shown 619 after fitting 620 :Not implemented. 630 highlight the current textcrtl and hide the error text control shown 631 after fitting 621 632 """ 622 633 return … … 624 635 def read_file(self, path): 625 636 """ 626 Read two columns file 627 @param path: the path to the file to read 637 Read two columns file 638 639 :param path: the path to the file to read 640 628 641 """ 629 642 try: … … 651 664 except: 652 665 raise 653 654 666 655 667 def createMemento(self): 656 668 """ 657 669 return the current state of the page 658 670 """ 659 671 return self.state.clone() … … 662 674 def save_current_state(self): 663 675 """ 664 676 Store current state 665 677 """ 666 678 ## save model option … … 743 755 def save_current_state_fit(self): 744 756 """ 745 757 Store current state for fit_page 746 758 """ 747 759 ## save model option … … 820 832 def check_invalid_panel(self): 821 833 """ 822 834 check if the user can already perform some action with this panel 823 835 """ 824 836 flag = False … … 829 841 wx.MessageBox(msg, 'Info') 830 842 return True 843 831 844 def reset_page_helper(self, state): 832 845 """ 833 Use page_state and change the state of existing page 834 @precondition: the page is already drawn or created 835 @postcondition: the state of the underlying data change as well as the 846 Use page_state and change the state of existing page 847 848 :precondition: the page is already drawn or created 849 850 :postcondition: the state of the underlying data change as well as the 836 851 state of the graphic interface 837 852 """ … … 958 973 def _selectDlg(self): 959 974 """ 960 975 open a dialog file to selected the customized dispersity 961 976 """ 962 977 import os … … 968 983 dlg.Destroy() 969 984 return path 970 971 985 972 986 def _reset_context_menu(self): 973 987 """ 974 988 reset the context menu 975 989 """ 976 990 for name, state in self.state.saved_states.iteritems(): … … 981 995 wx.EVT_MENU(self, id, self.onResetModel) 982 996 983 984 997 def _reset_plotting_range(self, state): 985 998 """ 986 999 Reset the plotting range to a given state 987 1000 """ 988 1001 if self.check_invalid_panel(): … … 993 1006 self.npts.SetValue(str(state.npts)) 994 1007 995 996 1008 def _save_typeOfmodel(self): 997 1009 """ 998 1010 save radiobutton containing the type model that can be selected 999 1011 """ 1000 1012 self.state.shape_rbutton = self.shape_rbutton.GetValue() … … 1010 1022 wx.PostEvent(self.parent, event) 1011 1023 1012 1013 1024 def _save_plotting_range(self ): 1014 1025 """ 1015 1026 save the state of plotting range 1016 1027 """ 1017 1028 self.state.qmin = self.qmin_x … … 1023 1034 def _onparamEnter_helper(self): 1024 1035 """ 1025 check if values entered by the user are changed and valid to replot 1026 model 1027 use : _check_value_enter 1036 check if values entered by the user are changed and valid to replot 1037 model 1028 1038 """ 1029 1039 # Flag to register when a parameter has changed. … … 1080 1090 def _update_paramv_on_fit(self): 1081 1091 """ 1082 1092 make sure that update param values just before the fitting 1083 1093 """ 1084 1094 #flag for qmin qmax check values … … 1145 1155 if self._is_2D():self.btEditMask.Enable(True) 1146 1156 1147 1148 1157 if not flag: 1149 1158 msg= "Cannot Plot or Fit :Must select a model or Fitting range is not valid!!! " … … 1156 1165 def _is_modified(self, is_modified): 1157 1166 """ 1158 1167 return to self._is_modified 1159 1168 """ 1160 1169 return is_modified … … 1162 1171 def _reset_parameters_state(self, listtorestore,statelist): 1163 1172 """ 1164 1173 Reset the parameters at the given state 1165 1174 """ 1166 1175 if len(statelist)==0 or len(listtorestore)==0 : … … 1207 1216 else: 1208 1217 item_page[6].Hide() 1209 1210 1218 1211 1219 def _copy_parameters_state(self, listtocopy, statelist): 1212 1220 """ 1213 copy the state of button 1214 @param listtocopy: the list of check button to copy 1215 @param statelist: list of state object to store the current state 1221 copy the state of button 1222 1223 :param listtocopy: the list of check button to copy 1224 :param statelist: list of state object to store the current state 1225 1216 1226 """ 1217 1227 if len(listtocopy)==0: … … 1255 1265 [min_state,min_value],[max_state , max_value],unit]) 1256 1266 1257 1258 1259 1267 def _set_model_sizer_selection(self, model): 1260 1268 """ 1261 1262 """ 1263 if model == None:1269 Display the sizer according to the type of the current model 1270 """ 1271 if model == None: 1264 1272 return 1265 1273 if hasattr(model ,"s_model"): … … 1368 1376 break 1369 1377 1370 1371 1378 def _draw_model(self): 1372 1379 """ 1373 Method to draw or refresh a plotted model. 1374 The method will use the data member from the model page 1375 to build a call to the fitting perspective manager. 1376 1377 [Note to coder: This way future changes will be done in only one place.] 1380 Method to draw or refresh a plotted model. 1381 The method will use the data member from the model page 1382 to build a call to the fitting perspective manager. 1378 1383 """ 1379 1384 if self.check_invalid_panel(): … … 1395 1400 def _set_model_sizer(self,sizer, box_sizer, title="", object=None): 1396 1401 """ 1397 1402 Use lists to fill a sizer for model info 1398 1403 """ 1399 1404 … … 1472 1477 self.SetScrollbars(20,20,25,65) 1473 1478 1474 1475 1479 def _show_combox(self, event): 1476 1480 """ 1477 1481 Show combox box associate with type of model selected 1478 1482 """ 1479 1483 if self.check_invalid_panel(): … … 1527 1531 def _populate_box(self, combobox, list): 1528 1532 """ 1529 fill combox box with dict item 1530 @param list: contains item to fill the combox 1533 fill combox box with dict item 1534 1535 :param list: contains item to fill the combox 1531 1536 item must model class 1532 1537 """ … … 1539 1544 name = model.name 1540 1545 combobox.Append(name,models) 1541 1542 1546 return 0 1543 1547 1544 #def _onparamEnter(self,event):1545 """1546 #when enter value on panel redraw model according to changed1547 """1548 """1549 tcrtl= event.GetEventObject()1550 1551 #Clear msg if previously shown.1552 msg= ""1553 wx.PostEvent(self.parent.parent, StatusEvent(status = msg ))1554 1555 ## save current state1556 self.save_current_state()1557 if event !=None:1558 #self._undo.Enable(True)1559 event = PageInfoEvent(page = self)1560 wx.PostEvent(self.parent, event)1561 1562 if check_float(tcrtl):1563 1564 self._onparamEnter_helper()1565 #event.Skip()1566 else:1567 msg= "Cannot Plot :Must enter a number!!! "1568 wx.PostEvent(self.parent.parent, StatusEvent(status = msg ))1569 #event.Skip()1570 return1571 """1572 1548 def _onQrangeEnter(self, event): 1573 1549 """ 1574 1550 Check validity of value enter in the Q range field 1575 1551 """ 1576 1552 … … 1631 1607 def _on_select_model_helper(self): 1632 1608 """ 1633 1609 call back for model selection 1634 1610 """ 1635 1611 ## reset dictionary containing reference to dispersion … … 1680 1656 def _validate_qrange(self, qmin_ctrl, qmax_ctrl): 1681 1657 """ 1682 Verify that the Q range controls have valid values 1683 and that Qmin < Qmax. 1684 1685 @param qmin_ctrl: text control for Qmin 1686 @param qmax_ctrl: text control for Qmax 1687 @return: True is the Q range is value, False otherwise 1658 Verify that the Q range controls have valid values 1659 and that Qmin < Qmax. 1660 1661 :param qmin_ctrl: text control for Qmin 1662 :param qmax_ctrl: text control for Qmax 1663 1664 :return: True is the Q range is value, False otherwise 1665 1688 1666 """ 1689 1667 qmin_validity = check_float(qmin_ctrl) … … 1712 1690 def _validate_Npts(self): 1713 1691 """ 1714 1715 1692 Validate the number of points for fitting is more than 10 points. 1693 If valid, setvalues Npts_fit otherwise post msg. 1716 1694 """ 1717 1695 #default flag … … 1743 1721 def _check_value_enter(self, list, modified): 1744 1722 """ 1745 @param list: model parameter and panel info 1746 each item of the list should be as follow: 1747 item=[cb state, name, value, "+/-", error of fit, min, max , units] 1723 :param list: model parameter and panel info 1724 :Note: each item of the list should be as follow: 1725 item=[check button state, parameter's name, 1726 paramater's value, string="+/-", 1727 parameter's error of fit, 1728 parameter's minimum value, 1729 parrameter's maximum value , 1730 parameter's units] 1748 1731 """ 1749 1732 is_modified = modified … … 1829 1812 def _set_dipers_Param(self, event): 1830 1813 """ 1831 1832 1833 1834 1835 1814 respond to self.enable_disp and self.disable_disp radio box. 1815 The dispersity object is reset inside the model into Gaussian. 1816 When the user select yes , this method display a combo box for more selection 1817 when the user selects No,the combo box disappears. 1818 Redraw the model with the default dispersity (Gaussian) 1836 1819 """ 1837 1820 if self.check_invalid_panel(): … … 1888 1871 def _layout_sizer_noDipers(self): 1889 1872 """ 1890 1873 Draw a sizer with no dispersity info 1891 1874 """ 1892 1875 ix=0 … … 1908 1891 def _reset_dispersity(self): 1909 1892 """ 1910 1893 put gaussian dispersity into current model 1911 1894 """ 1912 1895 if len(self.param_toFit)>0: … … 1950 1933 self.Refresh() 1951 1934 1952 1953 1935 def _on_select_Disp(self,event): 1954 1936 """ 1955 1956 1937 allow selecting different dispersion 1938 self.disp_list should change type later .now only gaussian 1957 1939 """ 1958 1940 n = self.disp_box.GetCurrentSelection() … … 1979 1961 def _set_sizer_arraydispersion(self): 1980 1962 """ 1981 1963 draw sizer with array dispersity parameters 1982 1964 """ 1983 1965 … … 2063 2045 wx.PostEvent(self.parent, event) 2064 2046 2065 2066 2067 2068 2069 2047 def _set_range_sizer(self, title, box_sizer=None, object1=None,object=None): 2070 2048 """ 2071 2049 Fill the Q range sizer 2072 2050 """ 2073 2051 #2D data? default … … 2144 2122 def _fill_save_sizer(self): 2145 2123 """ 2146 2124 Draw the layout for saving option 2147 2125 """ 2148 2126 self.sizer6.Clear(True) … … 2166 2144 def _lay_out(self): 2167 2145 """ 2168 returns self.Layout 2169 Note: Mac seems to like this better when self.Layout is called after fitting. 2146 returns self.Layout 2147 2148 :Note: Mac seems to like this better when self. 2149 Layout is called after fitting. 2170 2150 """ 2171 2151 self._sleep4sec() … … 2183 2163 def on_reset_clicked(self,event): 2184 2164 """ 2185 #On 'Reset' button for Q range clicked2165 On 'Reset' button for Q range clicked 2186 2166 """ 2187 2167 flag = True … … 2234 2214 self._onparamEnter_helper() 2235 2215 2236 2237 2216 self.save_current_state() 2238 2217 self.state.qmin = self.qmin_x … … 2247 2226 def on_model_help_clicked(self,event): 2248 2227 """ 2249 #On 'More details' button2228 on 'More details' button 2250 2229 """ 2251 2230 from help_panel import HelpWindow -
sansview/perspectives/fitting/console.py
re54d2c32 r5062bbf 22 22 If quiet is true, only print out final summary, not progress and 23 23 improvements. 24 25 :attr parent: the object that handle the messages 26 24 27 """ 25 28 self.parent= parent … … 96 99 97 100 def finalize(self): 101 """ 102 """ 98 103 if self.isbetter: 99 104 self.result.print_summary() 100 105 101 106 def abort(self): 107 """ 108 """ 102 109 if self.isbetter: 103 110 self.result.print_summary() 104 111 105 112 def update_fit(self, msg=""): 113 """ 114 """ 106 115 self.elapsed_time = time.time() - self.elapsed_time 107 116 msg = " Updating fit ...\n result:\n %s \n"%self.result.__str__() … … 110 119 111 120 def starting_fit(self): 121 """ 122 """ 112 123 wx.PostEvent(self.parent, StatusEvent(status="Starting the Fit...", 113 124 info="info",type="progress")) 114 125 115 126 def set_result(self, result): 127 """ 128 """ 116 129 self.result = result 117 130 -
sansview/perspectives/fitting/fit_thread.py
re54d2c32 r5062bbf 30 30 def isquit(self): 31 31 """ 32 @raise KeyboardInterrupt: when the thread is interrupted 32 :raise KeyboardInterrupt: when the thread is interrupted 33 33 34 """ 34 35 try: … … 39 40 def compute(self): 40 41 """ 41 42 Perform a fit 42 43 """ 43 44 try: … … 58 59 self.handler.error(msg=msg) 59 60 60 61 62 63 -
sansview/perspectives/fitting/fitpage.py
rcb190a19 r5062bbf 33 33 class FitPage(BasicPage): 34 34 """ 35 FitPanel class contains fields allowing to display results when 36 fitting a model and one data 37 @note: For Fit to be performed the user should check at least one parameter 35 FitPanel class contains fields allowing to display results when 36 fitting a model and one data 37 38 :note: For Fit to be performed the user should check at least one parameter 38 39 on fit Panel window. 39 40 40 """ 41 41 42 42 def __init__(self,parent, page_info): 43 """ 44 Initialization of the Panel 45 """ 43 46 BasicPage.__init__(self, parent, page_info) 44 45 """46 Initialization of the Panel47 """48 47 ## total number of point: float 49 48 self.npts=None … … 98 97 def _on_fit_complete(self, event): 99 98 """ 100 99 When fit is complete ,reset the fit button label. 101 100 """ 102 101 #self.btFit.SetLabel("Fit") … … 107 106 def _is_2D(self): 108 107 """ 109 Check if data_name is Data2D 110 @return: True or False 108 Check if data_name is Data2D 109 110 :return: True or False 111 111 112 """ 112 113 … … 117 118 def _on_engine_change(self, event): 118 119 """ 119 get an event containing the current name of the fit engine type 120 @param event: FitterTypeEvent containing the name of the current engine 120 get an event containing the current name of the fit engine type 121 122 :param event: FitterTypeEvent containing the name of the current engine 123 121 124 """ 122 125 self.engine_type = event.type … … 186 189 def _fill_range_sizer(self): 187 190 """ 188 189 191 Fill the sizer containing the plotting range 192 add access to npts 190 193 """ 191 194 is_2Ddata = False … … 231 234 self.smear_accuracy.SetToolTipString("'Higher' uses more Gaussian points for smearing computation.") 232 235 233 234 235 236 wx.EVT_COMBOBOX(self.smear_accuracy,-1, self._on_select_accuracy) 236 237 … … 390 391 def _fill_datainfo_sizer(self): 391 392 """ 392 393 fill sizer 0 with data info 393 394 """ 394 395 ## no loaded data , don't fill the sizer … … 455 456 def _fill_model_sizer(self, sizer): 456 457 """ 457 458 fill sizer containing model info 458 459 """ 459 460 ##Add model function Details button in fitpanel. … … 470 471 ## class base method to add view 2d button 471 472 self._set_model_sizer(sizer=sizer, box_sizer=boxsizer1, title="Model",object=self.model_help ) 472 473 473 474 474 def _set_sizer_dispersion(self, dispersity): 475 475 """ 476 476 draw sizer with gaussian dispersity parameters 477 477 """ 478 478 self.fittable_param=[] … … 749 749 ,None,None, None, None,None]) 750 750 751 752 751 self.state.disp_cb_dict = copy.deepcopy(self.disp_cb_dict) 753 752 … … 769 768 self.Layout() 770 769 771 772 770 def _onFit(self, event): 773 771 """ 774 772 Allow to fit 775 773 """ 776 774 #make sure all parameter values are updated. … … 796 794 return 797 795 798 799 796 self.select_param(event =None) 800 797 … … 820 817 # self.btFit.Bind(event= wx.EVT_BUTTON, handler=self._onFit, id=self.btFit.GetId()) 821 818 822 823 824 819 def _StopFit(self, event): 825 820 """ 826 821 Stop fit 827 822 """ 828 823 self.btFit.SetLabel("Fit") … … 835 830 def _on_select_model(self, event): 836 831 """ 837 832 call back for model selection 838 833 """ 839 834 self._on_select_model_helper() … … 879 874 event = PageInfoEvent(page = self) 880 875 wx.PostEvent(self.parent, event) 881 882 876 883 877 def _onparamEnter(self,event): 884 878 """ 885 879 when enter value on panel redraw model according to changed 886 880 """ 887 881 if self.model ==None: … … 941 935 def _onparamRangeEnter(self, event): 942 936 """ 943 937 Check validity of value enter in the parameters range field 944 938 """ 945 939 if self.check_invalid_panel(): … … 974 968 def _onQrangeEnter(self, event): 975 969 """ 976 970 Check validity of value enter in the Q range field 977 971 """ 978 972 if self.check_invalid_panel(): … … 1040 1034 wx.PostEvent(self.parent, event) 1041 1035 self.state_change= False 1042 1043 1036 return 1037 1044 1038 def _clear_Err_on_Fit(self): 1045 1039 """ 1046 1047 1040 hide the error text control shown 1041 after fitting 1048 1042 """ 1049 1043 … … 1084 1078 def _get_defult_custom_smear(self): 1085 1079 """ 1086 Get the defult values for custum smearing. 1087 """ 1088 1080 Get the defult values for custum smearing. 1081 """ 1089 1082 # get the default values 1090 1083 if self.dxl == None: self.dxl = 0.0 … … 1095 1088 def _get_smear_info(self): 1096 1089 """ 1097 Get the smear info from data. 1098 @return: self.smear_type, self.dq_l and self.dq_r, 1090 Get the smear info from data. 1091 1092 :return: self.smear_type, self.dq_l and self.dq_r, 1099 1093 respectively the type of the smear, dq_min and dq_max for pinhole smear data 1100 1094 while dxl and dxw for slit smear 1095 1101 1096 """ 1102 1097 … … 1118 1113 else: 1119 1114 return 1120 1121 1122 1123 1115 # check if it is pinhole smear and get min max if it is. 1124 1116 if data.dx != None and all(data.dx !=0): … … 1133 1125 self.dq_l = data.dxl[0] 1134 1126 if data.dxw != None and all(data.dxw !=0): 1135 self.dq_r = data.dxw[0] 1136 1127 self.dq_r = data.dxw[0] 1137 1128 #return self.smear_type,self.dq_l,self.dq_r 1138 1129 1139 1130 def _show_smear_sizer(self): 1140 1131 """ 1141 1132 Show only the sizers depending on smear selection 1142 1133 """ 1143 1134 # smear disabled … … 1190 1181 def _hide_all_smear_info(self): 1191 1182 """ 1192 1183 Hide all smearing messages in the set_smearer sizer 1193 1184 """ 1194 1185 self.smear_description_none.Hide() … … 1217 1208 def _set_accuracy_list(self): 1218 1209 """ 1219 1210 Set the list of an accuracy in 2D custum smear: Xhigh, High, Med, or Low 1220 1211 """ 1221 1212 # list of accuracy choices … … 1226 1217 def _on_select_accuracy(self,event): 1227 1218 """ 1228 1219 Select an accuracy in 2D custom smear: Xhigh, High, Med, or Low 1229 1220 """ 1230 1221 #event.Skip() 1231 1232 1222 # Check if the accuracy is same as before 1233 1234 1223 #self.smear2d_accuracy = event.GetEventObject().GetValue() 1235 1224 self.smear2d_accuracy = self.smear_accuracy.GetValue() … … 1239 1228 self.onSmear(event=None) 1240 1229 if self.current_smearer != None: 1241 self.current_smearer.set_accuracy(accuracy = self.smear2d_accuracy) 1242 1230 self.current_smearer.set_accuracy(accuracy = self.smear2d_accuracy) 1243 1231 event.Skip() 1244 1232 1245 1233 def _onMask(self, event): 1246 1234 """ 1247 1235 Build a panel to allow to edit Mask 1248 1236 """ 1249 1237 … … 1279 1267 def set_data(self, data): 1280 1268 """ 1281 1269 reset the current data 1282 1270 """ 1283 1271 self.data = data 1284 1285 1286 1272 if self.data is None: 1287 1273 data_min = "" … … 1302 1288 self.slit_smearer.Disable() 1303 1289 self.default_mask = copy.deepcopy(self.data.mask) 1304 1305 1290 1306 1291 self.formfactorbox.Enable() … … 1332 1317 self.btEditMask.Enable() 1333 1318 self.EditMask_title.Enable() 1334 1335 1336 1319 self.dataSource.SetValue(data_name) 1337 1320 self.qmin_x = data_min … … 1347 1330 self.state.qmax = self.qmax_x 1348 1331 1349 1350 1332 def reset_page(self, state,first=False): 1351 1333 """ 1352 1334 reset the state 1353 1335 """ 1354 1336 self.reset_page_helper(state) … … 1368 1350 def get_range(self): 1369 1351 """ 1370 1352 return the fitting range 1371 1353 """ 1372 1354 return float(self.qmin_x) , float(self.qmax_x) … … 1374 1356 def get_npts2fit(self): 1375 1357 """ 1376 return numbers of data points within qrange 1377 Note: This is for Park where chi2 is not normalized by Npts of fit 1358 return numbers of data points within qrange 1359 1360 :Note: This is for Park where chi2 is not normalized by Npts of fit 1361 1378 1362 """ 1379 1363 npts2fit = 0 … … 1393 1377 def set_npts2fit(self): 1394 1378 """ 1395 1379 setValue Npts for fitting 1396 1380 """ 1397 1381 self.Npts_fit.SetValue(str(self.get_npts2fit())) 1398 1382 1399 1400 1383 def get_chi2(self): 1401 1384 """ 1402 1385 return the current chi2 1403 1386 """ 1404 1387 return self.tcChi.GetValue() … … 1406 1389 def get_param_list(self): 1407 1390 """ 1408 @return self.param_toFit: list containing references to TextCtrl1391 :return self.param_toFit: list containing references to TextCtrl 1409 1392 checked.Theses TextCtrl will allow reference to parameters to fit. 1410 @raise: if return an empty list of parameter fit will nnote work 1393 1394 :raise: if return an empty list of parameter fit will nnote work 1411 1395 properly so raise ValueError,"missing parameter to fit" 1412 1396 """ … … 1418 1402 def onsetValues(self,chisqr,p_name, out,cov): 1419 1403 """ 1420 Build the panel from the fit result 1421 @param chisqr:Value of the goodness of fit metric 1422 @p_name: the name of parameters 1423 @param out:list of parameter with the best value found during fitting 1424 @param cov:Covariance matrix 1425 1404 Build the panel from the fit result 1405 1406 :param chisqr: Value of the goodness of fit metric 1407 :param p_name: the name of parameters 1408 :param out: list of parameter with the best value found during fitting 1409 :param cov: Covariance matrix 1410 1426 1411 """ 1427 1412 if out == None or not numpy.isfinite(chisqr): … … 1545 1530 def onPinholeSmear(self, event): 1546 1531 """ 1547 Create a custom pinhole smear object that will change the way residuals 1548 are compute when fitting 1549 @ accuracy: given by strings'High','Med', 'Low' FOR 2d, None for 1D 1532 Create a custom pinhole smear object that will change the way residuals 1533 are compute when fitting 1534 1535 :Note: accuracy is given by strings'High','Med', 'Low' FOR 2d, 1536 None for 1D 1537 1550 1538 """ 1551 1539 … … 1601 1589 def _is_changed_pinhole(self): 1602 1590 """ 1603 check if any of pinhole smear is changed 1604 return: True or False 1591 check if any of pinhole smear is changed 1592 1593 :return: True or False 1594 1605 1595 """ 1606 1596 # get the values … … 1627 1617 def _set_pinhole_smear(self): 1628 1618 """ 1629 Set custom pinhole smear 1630 return: msg 1619 Set custom pinhole smear 1620 1621 :return: msg 1622 1631 1623 """ 1632 1624 # copy data … … 1695 1687 def update_pinhole_smear(self): 1696 1688 """ 1697 called by kill_focus on pinhole TextCntrl 1698 to update the changes 1699 return: msg: False when wrong value was entered 1689 called by kill_focus on pinhole TextCntrl 1690 to update the changes 1691 1692 :return: False when wrong value was entered 1693 1700 1694 """ 1701 1695 # msg default … … 1714 1708 def onSlitSmear(self, event): 1715 1709 """ 1716 1717 1710 Create a custom slit smear object that will change the way residuals 1711 are compute when fitting 1718 1712 """ 1719 1713 … … 1770 1764 wx.PostEvent(self.manager.parent, StatusEvent(status = msg)) 1771 1765 1772 1773 1766 def _is_changed_slit(self): 1774 1767 """ 1775 check if any of slit lengths is changed 1776 return: True or False 1777 """ 1778 1768 check if any of slit lengths is changed 1769 1770 :return: True or False 1771 1772 """ 1779 1773 # get the values 1780 1774 width = self.smear_slit_width.GetValue() … … 1814 1808 def _set_slit_smear(self): 1815 1809 """ 1816 Set custom slit smear 1817 return: msg 1810 Set custom slit smear 1811 1812 :return: message to inform the user about the validity 1813 of the values entered for slit smear 1818 1814 """ 1819 1815 temp_smearer = None … … 1853 1849 ## set smearing value whether or not the data contain the smearing info 1854 1850 self.manager.set_smearer(smearer=self.current_smearer, qmin= float(self.qmin_x), qmax= float(self.qmax_x)) 1855 1856 1851 return msg 1857 1852 1858 1859 1853 def update_slit_smear(self): 1860 1854 """ 1861 called by kill_focus on pinhole TextCntrl 1862 to update the changes 1863 return: msg: False when wrong value was entered 1855 called by kill_focus on pinhole TextCntrl 1856 to update the changes 1857 1858 :return: False when wrong value was entered 1859 1864 1860 """ 1865 1861 # msg default … … 1876 1872 return True 1877 1873 1878 1879 1874 def onSmear(self, event): 1880 1875 """ 1881 1882 1876 Create a smear object that will change the way residuals 1877 are compute when fitting 1883 1878 """ 1884 1879 if event != None: … … 1950 1945 def on_complete_chisqr(self, event): 1951 1946 """ 1952 print result chisqr 1953 @event: activated by fitting/ complete after draw 1947 print result chisqr 1948 1949 :event: activated by fitting/ complete after draw 1950 1954 1951 """ 1955 1952 try: … … 1964 1961 pass 1965 1962 1966 1963 1967 1964 def select_all_param(self,event): 1968 1965 """ 1969 1966 set to true or false all checkBox given the main checkbox value cb1 1970 1967 """ 1971 1972 1968 self.param_toFit=[] 1973 1969 if self.parameters !=[]: … … 2008 2004 wx.PostEvent(self.parent, event) 2009 2005 2010 2011 2012 2006 def select_param(self,event): 2013 2007 """ 2014 2015 2008 Select TextCtrl checked for fitting purpose and stores them 2009 in self.param_toFit=[] list 2016 2010 """ 2017 2011 self.param_toFit=[] … … 2063 2057 event = PageInfoEvent(page = self) 2064 2058 wx.PostEvent(self.parent, event) 2065 2066 2059 2067 2068 2060 def set_model_param_sizer(self, model): 2069 2061 """ 2070 Build the panel from the model content 2071 @param model: the model selected in combo box for fitting purpose 2062 Build the panel from the model content 2063 2064 :param model: the model selected in combo box for fitting purpose 2065 2072 2066 """ 2073 2067 self.sizer3.Clear(True) … … 2087 2081 self.engine_type= self.manager._return_engine_type() 2088 2082 2089 2090 2083 box_description= wx.StaticBox(self, -1,str("Model Parameters")) 2091 2084 boxsizer1 = wx.StaticBoxSizer(box_description, wx.VERTICAL) … … 2326 2319 self.Refresh() 2327 2320 self.SetScrollbars(20,20,25,65) 2321 2328 2322 2329 2323 class BGTextCtrl(wx.TextCtrl): 2330 2324 """ 2331 2332 2333 2325 Text control used to display outputs. 2326 No editing allowed. The background is 2327 grayed out. User can't select text. 2334 2328 """ 2335 2329 def __init__(self, *args, **kwds): … … 2345 2339 def _click(self, event): 2346 2340 """ 2347 2348 2341 Prevent further handling of the mouse event 2342 by not calling Skip(). 2349 2343 """ 2350 2344 pass 2351 2352 class HelpWindow(wx.Frame): 2353 def __init__(self, parent, id, title): 2354 wx.Frame.__init__(self, parent, id, title, size=(570, 400)) 2355 2356 from sans.models.CylinderModel import CylinderModel 2357 model = CylinderModel() 2358 2359 from danse.common.plottools.plottables import Data1D 2360 data= Data1D(x=[1,2], y=[3,4], dy=[0.1, 0,1]) 2345 2346 """ 2347 Example: :: 2361 2348 2362 from fitpanel import PageInfo 2363 myinfo = PageInfo(self, model, data=data ) 2364 2365 ## add data 2366 2367 from models import ModelList 2368 mylist= ModelList() 2369 2370 from sans.models.SphereModel import SphereModel 2371 from sans.models.SquareWellStructure import SquareWellStructure 2372 from sans.models.DebyeModel import DebyeModel 2373 from sans.models.LineModel import LineModel 2374 name= "shapes" 2375 list1= [SphereModel] 2376 mylist.set_list( name, list1) 2377 2378 name= "Shape-independent" 2379 list1= [DebyeModel] 2380 mylist.set_list( name, list1) 2381 2382 name= "Structure Factors" 2383 list1= [SquareWellStructure] 2384 mylist.set_list( name, list1) 2385 2386 name= "Added models" 2387 list1= [LineModel] 2388 mylist.set_list( name, list1) 2389 2390 myinfo.model_list_box = mylist.get_list() 2391 2392 self.page = FitPage(self, myinfo) 2393 2394 self.Centre() 2395 self.Show(True) 2396 2397 if __name__=="__main__": 2398 app = wx.App() 2399 HelpWindow(None, -1, 'HelpWindow') 2400 app.MainLoop() 2401 2349 class HelpWindow(wx.Frame): 2350 def __init__(self, parent, id, title): 2351 wx.Frame.__init__(self, parent, id, title, size=(570, 400)) 2352 2353 from sans.models.CylinderModel import CylinderModel 2354 model = CylinderModel() 2355 2356 from danse.common.plottools.plottables import Data1D 2357 data= Data1D(x=[1,2], y=[3,4], dy=[0.1, 0,1]) 2358 2359 from fitpanel import PageInfo 2360 myinfo = PageInfo(self, model, data=data ) 2361 2362 ## add data 2363 2364 from models import ModelList 2365 mylist= ModelList() 2366 2367 from sans.models.SphereModel import SphereModel 2368 from sans.models.SquareWellStructure import SquareWellStructure 2369 from sans.models.DebyeModel import DebyeModel 2370 from sans.models.LineModel import LineModel 2371 name= "shapes" 2372 list1= [SphereModel] 2373 mylist.set_list( name, list1) 2374 2375 name= "Shape-independent" 2376 list1= [DebyeModel] 2377 mylist.set_list( name, list1) 2378 2379 name= "Structure Factors" 2380 list1= [SquareWellStructure] 2381 mylist.set_list( name, list1) 2382 2383 name= "Added models" 2384 list1= [LineModel] 2385 mylist.set_list( name, list1) 2386 2387 myinfo.model_list_box = mylist.get_list() 2388 2389 self.page = FitPage(self, myinfo) 2390 2391 self.Centre() 2392 self.Show(True) 2393 2394 if __name__=="__main__": 2395 app = wx.App() 2396 HelpWindow(None, -1, 'HelpWindow') 2397 app.MainLoop() 2398 """ 2399 -
sansview/perspectives/fitting/fitpanel.py
r31b0c47 r5062bbf 12 12 class StateIterator(object): 13 13 """ 14 15 16 17 14 Contains all saved state of a given page. 15 Provide position of the current state of a page, the first save state 16 and the last state for a given page. 17 Allow easy undo or redo for a given page 18 18 """ 19 19 def __init__(self): 20 """ 21 """ 20 22 self._current=0 21 23 22 23 24 def __iter__(self): 25 """ 26 """ 24 27 return self 25 28 26 27 29 def first(self): 30 """ 31 """ 28 32 self._current =0 29 33 return self._current 30 34 31 35 def next(self, max ): 36 """ 37 """ 32 38 if self._current < max: 33 39 self._current += 1 … … 35 41 36 42 def previous(self): 43 """ 44 """ 37 45 if self._current > 0: 38 46 self._current = self._current -1 … … 40 48 41 49 def currentPosition(self): 50 """ 51 """ 42 52 return self._current 43 53 44 54 def setPosition(self, value): 55 """ 56 """ 45 57 if value >=0: 46 58 self._current = int(value) 47 59 48 60 49 class ListOfState(list): 61 class ListOfState(list): 62 """ 63 """ 50 64 def __init__(self, *args, **kw): 51 65 list.__init__(self, *args, **kw) … … 53 67 54 68 def appendItem(self, x): 69 """ 70 """ 55 71 self.append(x) 56 72 self.iterator.setPosition(value= len(self)-1) 57 73 58 74 def removeItem(self, x): 75 """ 76 """ 59 77 self.iterator.previous() 60 78 self.remove(x) 61 79 62 80 def getPreviousItem(self): 81 """ 82 """ 63 83 position = self.iterator.previous() 64 84 … … 69 89 70 90 def getNextItem(self): 91 """ 92 """ 71 93 position = self.iterator.next(max= len(self)-1) 72 94 if position >= len(self): … … 76 98 77 99 def getCurrentItem(self): 100 """ 101 """ 78 102 postion = self.iterator.currentPosition() 79 103 if postion >= 0 and position < len(self): … … 83 107 84 108 def getCurrentPosition(self): 109 """ 110 """ 85 111 return self.iterator.currentPosition() 86 112 … … 88 114 class PageInfo(object): 89 115 """ 90 91 116 this class contains the minimum numbers of data members 117 a fitpage or model page need to be initialized. 92 118 """ 93 119 data = None … … 106 132 event_owner=None, model_list_box=None, name=None): 107 133 """ 108 134 Initialize data members 109 135 """ 110 136 self.data = data … … 121 147 122 148 """ 123 FitPanel class contains fields allowing to fit models and data 124 @note: For Fit to be performed the user should check at least one parameter 149 FitPanel class contains fields allowing to fit models and data 150 151 :note: For Fit to be performed the user should check at least one parameter 125 152 on fit Panel window. 126 153 … … 133 160 134 161 def __init__(self, parent, *args, **kwargs): 162 """ 163 """ 135 164 wx.aui.AuiNotebook.__init__(self, parent, -1, 136 165 style= wx.aui.AUI_NB_WINDOWLIST_BUTTON| … … 173 202 def set_state(self, state): 174 203 """ 175 204 Restore state of the panel 176 205 """ 177 206 page_is_opened = False … … 193 222 def on_close_page(self, event): 194 223 """ 195 224 close page and remove all references to the closed page 196 225 """ 197 226 nbr_page = self.GetPageCount() … … 204 233 def close_page_with_data(self, deleted_data): 205 234 """ 206 235 close a fit page when its data is completely remove from the graph 207 236 """ 208 237 if deleted_data is None: … … 223 252 def set_manager(self, manager): 224 253 """ 225 set panel manager 226 @param manager: instance of plugin fitting 254 set panel manager 255 256 :param manager: instance of plugin fitting 257 227 258 """ 228 259 self.manager = manager … … 231 262 def set_owner(self,owner): 232 263 """ 233 set and owner for fitpanel 234 @param owner: the class responsible of plotting 264 set and owner for fitpanel 265 266 :param owner: the class responsible of plotting 267 235 268 """ 236 269 self.event_owner = owner … … 238 271 def set_model_list(self, dict): 239 272 """ 240 copy a dictionary of model into its own dictionary 241 @param dict: dictionnary made of model name as key and model class 273 copy a dictionary of model into its own dictionary 274 275 :param dict: dictionnary made of model name as key and model class 242 276 as value 243 277 """ 244 278 self.model_list_box = dict 245 279 246 247 280 def get_current_page(self): 248 281 """ 249 @return the current page selected 282 :return: the current page selected 283 250 284 """ 251 285 return self.GetPage(self.GetSelection() ) … … 253 287 def add_sim_page(self): 254 288 """ 255 289 Add the simultaneous fit page 256 290 """ 257 291 from simfitpage import SimultaneousFitPage … … 265 299 def get_page_info(self, data=None): 266 300 """ 267 301 fill information required to add a page in the fit panel 268 302 """ 269 303 name = "Fit Page" … … 290 324 def add_empty_page(self): 291 325 """ 292 326 add an empty page 293 327 """ 294 328 page_info = self.get_page_info() … … 301 335 def add_page(self, page_info): 302 336 """ 303 337 add a new page 304 338 """ 305 339 from fitpage import FitPage … … 312 346 def change_page_content(self, data, index): 313 347 """ 314 348 replace the contains of an existing page 315 349 """ 316 350 page_info = self.get_page_info(data=data) … … 328 362 def replace_page(self, index, page_info, type): 329 363 """ 330 364 replace an existing page 331 365 """ 332 366 self.DeletePage(index) … … 336 370 def add_fit_page(self, data, reset=False): 337 371 """ 338 Add a fitting page on the notebook contained by fitpanel 339 @param data: data to fit 340 @return panel : page just added for further used. is used by fitting module 372 Add a fitting page on the notebook contained by fitpanel 373 374 :param data: data to fit 375 376 :return panel : page just added for further used. is used by fitting module 377 341 378 """ 342 379 if data is None: … … 381 418 def _onGetstate(self, event): 382 419 """ 383 420 copy the state of a page 384 421 """ 385 422 page= event.page … … 389 426 def _onUndo(self, event ): 390 427 """ 391 428 return the previous state of a given page is available 392 429 """ 393 430 page = event.page … … 402 439 def _onRedo(self, event): 403 440 """ 404 441 return the next state available 405 442 """ 406 443 page = event.page … … 417 454 def _close_helper(self, selected_page): 418 455 """ 419 456 Delete the given page from the notebook 420 457 """ 421 458 #remove hint page -
sansview/perspectives/fitting/fitproblem.py
r2f189dc r5062bbf 2 2 class FitProblem: 3 3 """ 4 5 6 4 FitProblem class allows to link a model with the new name created in _on_model, 5 a name theory created with that model and the data fitted with the model. 6 FitProblem is mostly used as value of the dictionary by fitting module. 7 7 """ 8 9 8 def __init__(self): 10 11 9 """ 12 10 contains information about data and model to fit 13 11 """ 14 12 ## data used for fitting … … 33 31 def clone(self): 34 32 """ 35 33 copy fitproblem 36 34 """ 37 35 import copy … … 49 47 obj.qmin = copy.deepcopy(self.qmin) 50 48 obj.qmax = copy.deepcopy(self.qmax) 51 52 49 return obj 53 54 50 55 51 def set_smearer(self, smearer): 56 52 """ 57 save reference of smear object on fitdata 58 @param smear : smear object from DataLoader 53 save reference of smear object on fitdata 54 55 :param smear: smear object from DataLoader 56 59 57 """ 60 58 self.smearer= smearer … … 62 60 def get_smearer(self): 63 61 """ 64 62 return smear object 65 63 """ 66 64 return self.smearer 67 65 68 def save_model_name(self, name): 66 def save_model_name(self, name): 67 """ 68 """ 69 69 self.name_per_page= name 70 70 71 72 71 def get_name(self): 72 """ 73 """ 73 74 return self.name_per_page 74 75 75 76 def set_model(self,model): 76 77 """ 77 associates each model with its new created name 78 @param model: model selected 79 @param name: name created for model 78 associates each model with its new created name 79 80 :param model: model selected 81 :param name: name created for model 82 80 83 """ 81 84 self.model= model 82 85 86 def get_model(self): 87 """ 88 :return: saved model 83 89 84 def get_model(self): 85 """ @return: saved model """ 90 """ 86 91 return self.model 87 92 88 93 def set_index(self, index): 89 94 """ 90 95 set index of the model name 91 96 """ 92 97 self.model_index = index … … 94 99 def get_index(self): 95 100 """ 96 101 get index of the model name 97 102 """ 98 103 return self.model_index … … 100 105 def add_plotted_data(self,data): 101 106 """ 102 save a copy of the data select to fit 103 @param data: data selected 107 save a copy of the data select to fit 108 109 :param data: data selected 110 104 111 """ 105 112 self.plotted_data = data … … 107 114 108 115 def get_plotted_data(self): 109 """ @return: list of data dList""" 116 """ 117 :return: list of data dList 118 119 """ 110 120 return self.plotted_data 111 112 121 113 122 def add_fit_data(self,data): 114 123 """ 115 save a copy of the data select to fit 116 @param data: data selected 124 save a copy of the data select to fit 125 126 :param data: data selected 127 117 128 """ 118 129 self.fit_data = data 119 130 120 121 131 def get_fit_data(self): 132 """ 133 """ 122 134 return self.fit_data 123 135 124 125 136 def set_model_param(self,name,value=None): 126 137 """ 127 Store the name and value of a parameter of this fitproblem's model 128 @param name: name of the given parameter 129 @param value: value of that parameter 138 Store the name and value of a parameter of this fitproblem's model 139 140 :param name: name of the given parameter 141 :param value: value of that parameter 142 130 143 """ 131 144 self.list_param.append([name,value]) 132 145 133 134 146 def get_model_param(self): 135 147 """ 136 @return list of couple of parameter name and value148 return list of couple of parameter name and value 137 149 """ 138 150 return self.list_param 139 151 140 141 152 def schedule_tofit(self, schedule=0): 142 153 """ 143 154 set schedule to true to decide if this fit must be performed 144 155 """ 145 156 self.schedule=schedule 146 157 147 148 158 def get_scheduled(self): 149 """ return true or false if a problem as being schedule for fitting""" 159 """ 160 return true or false if a problem as being schedule for fitting 161 """ 150 162 return self.schedule 151 163 152 153 164 def set_range(self, qmin=None, qmax=None): 154 165 """ 155 166 set fitting range 156 167 """ 157 168 self.qmin = qmin 158 169 self.qmax = qmax 159 170 160 161 171 def get_range(self): 162 172 """ 163 @return fitting range: 173 :return: fitting range 174 164 175 """ 165 176 return self.qmin, self.qmax … … 170 181 """ 171 182 self.list_param=[] 172 173 174 175 183 -
sansview/perspectives/fitting/fitting.py
r2882fb5 r5062bbf 1 """ 2 This software was developed by the University of Tennessee as part of the 3 Distributed Data Analysis of Neutron Scattering Experiments (DANSE) 4 project funded by the US National Science Foundation. 5 6 See the license text in license.txt 7 8 copyright 2009, University of Tennessee 9 """ 1 2 3 ################################################################################ 4 #This software was developed by the University of Tennessee as part of the 5 #Distributed Data Analysis of Neutron Scattering Experiments (DANSE) 6 #project funded by the US National Science Foundation. 7 # 8 #See the license text in license.txt 9 # 10 #copyright 2009, University of Tennessee 11 ################################################################################ 12 13 10 14 import re 11 15 import sys … … 49 53 50 54 from fitpage import Chi2UpdateEvent 55 56 51 57 class PlotInfo: 52 58 """ 53 59 store some plotting field 54 60 """ 55 61 _xunit = 'A^{-1}' … … 65 71 class Plugin: 66 72 """ 67 73 Fitting plugin is used to perform fit 68 74 """ 69 75 def __init__(self): 76 """ 77 """ 70 78 ## Plug-in name 71 79 self.sub_menu = "Fitting" … … 114 122 def populate_menu(self, id, owner): 115 123 """ 116 Create a menu for the Fitting plug-in 117 @param id: id to create a menu 118 @param owner: owner of menu 119 @ return : list of information to populate the main menu 124 Create a menu for the Fitting plug-in 125 126 :param id: id to create a menu 127 :param owner: owner of menu 128 129 :return: list of information to populate the main menu 130 120 131 """ 121 132 #Menu for fitting … … 157 168 def on_add_sim_page(self, event): 158 169 """ 159 170 Create a page to access simultaneous fit option 160 171 """ 161 172 Plugin.on_perspective(self,event=event) … … 169 180 def help(self, evt): 170 181 """ 171 Show a general help dialog. 172 TODO: replace the text with a nice image 182 Show a general help dialog. 173 183 """ 174 184 … … 177 187 frame.Show(True) 178 188 179 180 189 def get_context_menu(self, graph=None): 181 190 """ 182 Get the context menu items available for P(r).them allow fitting option 183 for Data2D and Data1D only. 184 185 @param graph: the Graph object to which we attach the context menu 186 @return: a list of menu items with call-back function 187 @note: if Data1D was generated from Theory1D 188 the fitting option is not allowed 191 Get the context menu items available for P(r).them allow fitting option 192 for Data2D and Data1D only. 193 194 :param graph: the Graph object to which we attach the context menu 195 196 :return: a list of menu items with call-back function 197 198 :note: if Data1D was generated from Theory1D 199 the fitting option is not allowed 200 189 201 """ 190 202 self.graph = graph … … 221 233 def get_panels(self, parent): 222 234 """ 223 235 Create and return a list of panel objects 224 236 """ 225 237 self.parent = parent … … 252 264 def get_perspective(self): 253 265 """ 254 266 Get the list of panel names for this perspective 255 267 """ 256 268 return self.perspective 257 269 258 259 270 def on_perspective(self, event): 260 271 """ 261 262 263 272 Call back function for the perspective menu item. 273 We notify the parent window that the perspective 274 has changed. 264 275 """ 265 276 self.parent.set_perspective(self.perspective) … … 267 278 def set_default_perspective(self): 268 279 """ 269 270 271 272 280 Call back method that True to notify the parent that the current plug-in 281 can be set as default perspective. 282 when returning False, the plug-in is not candidate for an automatic 283 default perspective setting 273 284 """ 274 285 return True … … 276 287 def post_init(self): 277 288 """ 278 289 Post initialization call back to close the loose ends 279 290 """ 280 291 pass … … 282 293 def set_state(self, state, datainfo=None): 283 294 """ 284 Call-back method for the fit page state reader. 285 This method is called when a .fitv file is loaded. 286 @param state: PageState object 295 Call-back method for the fit page state reader. 296 This method is called when a .fitv file is loaded. 297 298 :param state: PageState object 299 287 300 """ 288 301 #working on reading state … … 299 312 def save_fit_state(self, filepath, fitstate): 300 313 """ 301 314 save fit page state into file 302 315 """ 303 316 self.state_reader.write(filename=filepath, fitstate=fitstate) … … 305 318 def copy_data(self, item, dy=None): 306 319 """ 307 308 309 @param return320 receive a data 1D and the list of errors on dy 321 and create a new data1D data 322 310 323 """ 311 324 id = None … … 327 340 def set_fit_range(self, page, qmin, qmax): 328 341 """ 329 342 Set the fitting range of a given page 330 343 """ 331 344 if page in self.page_finder.iterkeys(): … … 335 348 def schedule_for_fit(self,value=0,page=None,fitproblem =None): 336 349 """ 337 Set the fit problem field to 0 or 1 to schedule that problem to fit. 338 Schedule the specified fitproblem or get the fit problem related to 339 the current page and set value. 340 @param value : integer 0 or 1 341 @param fitproblem: fitproblem to schedule or not to fit 350 Set the fit problem field to 0 or 1 to schedule that problem to fit. 351 Schedule the specified fitproblem or get the fit problem related to 352 the current page and set value. 353 354 :param value: integer 0 or 1 355 :param fitproblem: fitproblem to schedule or not to fit 356 342 357 """ 343 358 if fitproblem !=None: … … 349 364 350 365 def get_page_finder(self): 351 """ @return self.page_finder used also by simfitpage.py""" 366 """ 367 return self.page_finder used also by simfitpage.py 368 """ 352 369 return self.page_finder 353 370 354 355 371 def set_page_finder(self,modelname,names,values): 356 372 """ 357 Used by simfitpage.py to reset a parameter given the string constrainst. 358 @param modelname: the name ot the model for with the parameter has to reset 359 @param value: can be a string in this case. 360 @param names: the paramter name 361 @note: expecting park used for fit. 373 Used by simfitpage.py to reset a parameter given the string constrainst. 374 375 :param modelname: the name ot the model for with the parameter has to reset 376 :param value: can be a string in this case. 377 :param names: the paramter name 378 379 :note: expecting park used for fit. 380 362 381 """ 363 382 sim_page= self.sim_page … … 372 391 def split_string(self,item): 373 392 """ 374 receive a word containing dot and split it. used to split parameterset 375 name into model name and parameter name example: 393 receive a word containing dot and split it. used to split parameterset 394 name into model name and parameter name example: :: 395 376 396 paramaterset (item) = M1.A 377 @return model_name =M1 , parameter name =A 397 Will return model_name = M1 , parameter name = A 398 378 399 """ 379 400 if string.find(item,".")!=-1: … … 389 410 def stop_fit(self): 390 411 """ 391 412 Stop the fit engine 392 413 """ 393 414 if self.calc_fit!= None and self.calc_fit.isrunning(): … … 398 419 def set_smearer_nodraw(self,smearer, qmin=None, qmax=None): 399 420 """ 400 Get a smear object and store it to a fit problem 401 @param smearer: smear object to allow smearing data 421 Get a smear object and store it to a fit problem 422 423 :param smearer: smear object to allow smearing data 424 402 425 """ 403 426 self.current_pg=self.fit_panel.get_current_page() … … 408 431 ## if user has already selected a model to plot 409 432 ## redraw the model with data smeared 410 411 433 smear =self.page_finder[self.current_pg].get_smearer() 412 434 413 435 def set_smearer(self,smearer, qmin=None, qmax=None): 414 436 """ 415 Get a smear object and store it to a fit problem 416 @param smearer: smear object to allow smearing data 437 Get a smear object and store it to a fit problem 438 439 :param smearer: smear object to allow smearing data 440 417 441 """ 418 442 self.current_pg=self.fit_panel.get_current_page() … … 433 457 qmin= DEFAULT_QMIN, qmax= DEFAULT_QMAX, qstep= DEFAULT_NPTS): 434 458 """ 435 Draw model. 436 @param model: the model to draw 437 @param name: the name of the model to draw 438 @param data: the data on which the model is based to be drawn 439 @param description: model's description 440 @param enable1D: if true enable drawing model 1D 441 @param enable2D: if true enable drawing model 2D 442 @param qmin: Range's minimum value to draw model 443 @param qmax: Range's maximum value to draw model 444 @param qstep: number of step to divide the x and y-axis 459 Draw model. 460 461 :param model: the model to draw 462 :param name: the name of the model to draw 463 :param data: the data on which the model is based to be drawn 464 :param description: model's description 465 :param enable1D: if true enable drawing model 1D 466 :param enable2D: if true enable drawing model 2D 467 :param qmin: Range's minimum value to draw model 468 :param qmax: Range's maximum value to draw model 469 :param qstep: number of step to divide the x and y-axis 445 470 446 471 """ 447 448 472 if data.__class__.__name__ !="Data2D": 449 473 ## draw model 1D with no loaded data … … 464 488 def onFit(self): 465 489 """ 466 490 perform fit 467 491 """ 468 492 ## count the number of fitproblem schedule to fit … … 545 569 def remove_plot(self, page, theory=False): 546 570 """ 547 571 remove model plot when a fit page is closed 548 572 """ 549 573 fitproblem = self.page_finder[page] … … 570 594 def create_fittable_data2D(self, data): 571 595 """ 572 check if the current data is a data 2d and add dy to that data 573 @return Data2D 596 check if the current data is a data 2d and add dy to that data 597 598 :return: Data2D 599 574 600 """ 575 601 if data.__class__.__name__ != "Data2D": … … 591 617 def create_fittable_data1D(self, data): 592 618 """ 593 check if the current data is a theory 1d and add dy to that data 594 @return Data1D 619 check if the current data is a theory 1d and add dy to that data 620 621 :return: Data1D 622 595 623 """ 596 624 class_name = data.__class__.__name__ … … 621 649 def store_page(self, page, data): 622 650 """ 623 Helper to save page reference into the plug-in 624 @param page: page to store 651 Helper to save page reference into the plug-in 652 653 :param page: page to store 654 625 655 """ 626 656 page.set_data(data) … … 632 662 def add_fit_page(self, data): 633 663 """ 634 635 664 given a data, ask to the fitting panel to create a new fitting page, 665 get this page and store it into the page_finder of this plug-in 636 666 """ 637 667 try: … … 651 681 def _onEVT_SLICER_PANEL(self, event): 652 682 """ 653 receive and event telling to update a panel with a name starting with 654 event.panel_name. this method update slicer panel for a given interactor. 655 @param event: contains type of slicer , paramaters for updating the panel 683 receive and event telling to update a panel with a name starting with 684 event.panel_name. this method update slicer panel for a given interactor. 685 686 :param event: contains type of slicer , paramaters for updating the panel 656 687 and panel_name to find the slicer 's panel concerned. 657 688 """ … … 662 693 self.parent._mgr.Update() 663 694 664 665 695 def _closed_fitpage(self, event): 666 696 """ 667 668 697 request fitpanel to close a given page when its unique data is removed 698 from the plot. close fitpage only when the a loaded data is removed 669 699 """ 670 700 if event is None or event.data is None: … … 677 707 def _add_page_onmenu(self, name,fitproblem=None): 678 708 """ 679 709 Add name of a closed page of fitpanel in a menu 680 710 """ 681 711 list = self.menu1.GetMenuItems() … … 691 721 wx.EVT_MENU(self.parent,event_id, self._open_closed_page) 692 722 693 694 723 def _open_closed_page(self, event): 695 724 """ 696 725 reopen a closed page 697 726 """ 698 727 for name, value in self.closed_page_dict.iteritems(): … … 712 741 reset= True) 713 742 break 714 715 743 716 744 def _reset_schedule_problem(self, value=0): 717 745 """ 718 746 unschedule or schedule all fitproblem to be fit 719 747 """ 720 748 for page, fitproblem in self.page_finder.iteritems(): … … 723 751 def _fit_helper(self,pars,value, id, title="Single Fit " ): 724 752 """ 725 753 helper for fitting 726 754 """ 727 755 metadata = value.get_fit_data() … … 761 789 def _onSelect(self,event): 762 790 """ 763 764 791 when Select data to fit a new page is created .Its reference is 792 added to self.page_finder 765 793 """ 766 794 self.panel = event.GetEventObject() … … 778 806 def _single_fit_completed(self,result,pars,cpage, elapsed=None): 779 807 """ 780 Display fit result on one page of the notebook. 781 @param result: result of fit 782 @param pars: list of names of parameters fitted 783 @param current_pg: the page where information will be displayed 784 @param qmin: the minimum value of x to replot the model 785 @param qmax: the maximum value of x to replot model 808 Display fit result on one page of the notebook. 809 810 :param result: result of fit 811 :param pars: list of names of parameters fitted 812 :param current_pg: the page where information will be displayed 813 :param qmin: the minimum value of x to replot the model 814 :param qmax: the maximum value of x to replot model 786 815 787 816 """ … … 813 842 return 814 843 815 816 844 def _simul_fit_completed(self,result,pars=None,cpage=None, elapsed=None): 817 845 """ 818 Parameter estimation completed, 819 display the results to the user 820 @param alpha: estimated best alpha 821 @param elapsed: computation time 846 Parameter estimation completed, 847 display the results to the user 848 849 :param alpha: estimated best alpha 850 :param elapsed: computation time 851 822 852 """ 823 853 ## fit more than 1 model at the same time … … 860 890 return 861 891 862 863 864 892 def _on_show_panel(self, event): 865 print "_on_show_panel: fitting" 866 867 893 """ 894 """ 895 #print "_on_show_panel: fitting" 896 pass 897 868 898 def _onset_engine_park(self,event): 869 899 """ 870 900 set engine to park 871 901 """ 872 902 Plugin.on_perspective(self,event=event) 873 903 self._on_change_engine('park') 874 904 875 876 905 def _onset_engine_scipy(self,event): 877 906 """ 878 907 set engine to scipy 879 908 """ 880 909 self._on_change_engine('scipy') … … 882 911 def _on_slicer_event(self, event): 883 912 """ 884 Receive a panel as event and send it to guiframe 885 @param event: event containing a panel 886 """ 887 888 if event.panel!=None: 913 Receive a panel as event and send it to guiframe 914 915 :param event: event containing a panel 916 917 """ 918 if event.panel is not None: 889 919 new_panel = event.panel 890 920 self.slicer_panels.append(event.panel) … … 897 927 new_panel.uid = event_id 898 928 self.mypanels.append(new_panel) 899 return 900 929 901 930 def _onclearslicer(self, event): 902 931 """ 903 932 Clear the boxslicer when close the panel associate with this slicer 904 933 """ 905 934 name =event.GetPane().caption … … 915 944 break 916 945 break 917 918 919 920 946 921 947 def _return_engine_type(self): 922 948 """ 923 949 return the current type of engine 924 950 """ 925 951 return self._fit_engine … … 928 954 def _on_change_engine(self, engine='park'): 929 955 """ 930 Allow to select the type of engine to perform fit 931 @param engine: the key work of the engine 932 """ 933 956 Allow to select the type of engine to perform fit 957 958 :param engine: the key work of the engine 959 960 """ 934 961 ## saving fit engine name 935 962 self._fit_engine = engine … … 952 979 wx.PostEvent(key, event) 953 980 954 955 981 def _on_model_panel(self, evt): 956 982 """ 957 react to model selection on any combo box or model menu.plot the model 958 @param evt: wx.combobox event 983 react to model selection on any combo box or model menu.plot the model 984 985 :param evt: wx.combobox event 986 959 987 """ 960 988 model = evt.model … … 997 1025 def _on_model_menu(self, evt): 998 1026 """ 999 Plot a theory from a model selected from the menu 1000 @param evt: wx.menu event 1027 Plot a theory from a model selected from the menu 1028 1029 :param evt: wx.menu event 1030 1001 1031 """ 1002 1032 model = evt.model … … 1006 1036 # the content will be updated and the plot refreshed 1007 1037 self.fit_panel.add_model_page(model,topmenu=True) 1008 1009 1010 1011 1038 1012 1039 def _update1D(self,x, output): 1013 1040 """ 1014 1041 Update the output of plotting model 1D 1015 1042 """ 1016 1043 wx.PostEvent(self.parent, StatusEvent(status="Plot \ … … 1019 1046 #self.calc_thread.ready(0.01) 1020 1047 1021 1022 1048 def _fill_default_model2D(self, theory, qmax,qstep, qmin=None): 1023 1049 """ 1024 fill Data2D with default value 1025 @param theory: Data2D to fill 1050 fill Data2D with default value 1051 1052 :param theory: Data2D to fill 1053 1026 1054 """ 1027 1055 from DataLoader.data_info import Detector, Source … … 1039 1067 theory.detector[0].beam_center.x= qmax 1040 1068 theory.detector[0].beam_center.y= qmax 1041 1042 1069 1043 1070 ## create x_bins and y_bins of the model 2D 1044 1071 pixel_width_x = theory.detector[0].pixel_size.x … … 1100 1127 theory.ymin= ymin 1101 1128 theory.ymax= ymax 1102 theory.group_id ="Model" 1103 theory.id ="Model" 1104 1129 theory.group_id = "Model" 1130 theory.id = "Model" 1105 1131 1106 1132 def _get_plotting_info(self, data=None): 1107 1133 """ 1108 get plotting info from data if data !=None 1109 else use some default 1134 get plotting info from data if data !=None else use some default 1110 1135 """ 1111 1136 my_info = PlotInfo() … … 1125 1150 if hasattr(data, "group_id"): 1126 1151 my_info.group_id= data.group_id 1127 1128 1152 return my_info 1129 1153 1130 1131 1154 def _complete1D(self, x,y, elapsed,index,model,data=None): 1132 1155 """ 1133 1156 Complete plotting 1D data 1134 1157 """ 1135 1158 try: … … 1169 1192 msg+= " %s"%sys.exc_value 1170 1193 wx.PostEvent( self.parent, StatusEvent(status= msg, type="stop")) 1171 return 1172 1194 1173 1195 def _update2D(self, output,time=None): 1174 1196 """ 1175 1197 Update the output of plotting model 1176 1198 """ 1177 1199 wx.PostEvent(self.parent, StatusEvent(status="Plot \ … … 1179 1201 self.ready_fit() 1180 1202 #self.calc_thread.ready(0.01) 1181 1182 1183 def _complete2D(self, image,data, model, elapsed,index,qmin, qmax,qstep=DEFAULT_NPTS):1184 """ 1185 1186 1203 1204 def _complete2D(self, image, data, model, elapsed, index, qmin, 1205 qmax, qstep=DEFAULT_NPTS): 1206 """ 1207 Complete get the result of modelthread and create model 2D 1208 that can be plot. 1187 1209 """ 1188 1210 err_image = numpy.zeros(numpy.shape(image)) … … 1192 1214 1193 1215 if data ==None: 1194 self._fill_default_model2D(theory= theory, qmax=qmax,qstep=qstep, qmin= qmin) 1216 self._fill_default_model2D(theory=theory, qmax=qmax, qstep=qstep, 1217 qmin= qmin) 1195 1218 1196 1219 else: … … 1218 1241 title="Analytical model 2D ", reset=True )) 1219 1242 # Chisqr in fitpage 1220 current_pg=self.fit_panel.get_current_page() 1221 wx.PostEvent(current_pg,Chi2UpdateEvent(output=self._cal_chisqr(data=data,index=index))) 1243 current_pg = self.fit_panel.get_current_page() 1244 wx.PostEvent(current_pg, 1245 Chi2UpdateEvent(output=self._cal_chisqr(data=data,index=index))) 1222 1246 msg = "Plot 2D complete !" 1223 wx.PostEvent( self.parent, StatusEvent( status= msg, type="stop" ))1247 wx.PostEvent( self.parent, StatusEvent(status=msg, type="stop" )) 1224 1248 1225 1249 def _on_data_error(self, event): 1226 1250 """ 1227 1228 1251 receives and event from plotting plu-gins to store the data name and 1252 their errors of y coordinates for 1Data hide and show error 1229 1253 """ 1230 1254 self.err_dy = event.err_dy … … 1233 1257 qmin=DEFAULT_QMIN, qmax=DEFAULT_QMAX, qstep=DEFAULT_NPTS): 1234 1258 """ 1235 draw model in 2D 1236 @param model: instance of the model to draw 1237 @param description: the description of the model 1238 @param enable2D: when True allows to draw model 2D 1239 @param qmin: the minimum value to draw model 2D 1240 @param qmax: the maximum value to draw model 2D 1241 @param qstep: the number of division of Qx and Qy of the model to draw 1259 draw model in 2D 1260 1261 :param model: instance of the model to draw 1262 :param description: the description of the model 1263 :param enable2D: when True allows to draw model 2D 1264 :param qmin: the minimum value to draw model 2D 1265 :param qmax: the maximum value to draw model 2D 1266 :param qstep: the number of division of Qx and Qy of the model to draw 1242 1267 1243 1268 """ … … 1283 1308 wx.PostEvent( self.parent, StatusEvent(status= msg )) 1284 1309 1285 1286 1310 def _draw_model1D(self, model, data=None, smearer= None, 1287 1311 qmin=DEFAULT_QMIN, qmax=DEFAULT_QMAX, qstep= DEFAULT_NPTS,enable1D= True): 1288 1312 """ 1289 Draw model 1D from loaded data1D 1290 @param data: loaded data 1291 @param model: the model to plot 1313 Draw model 1D from loaded data1D 1314 1315 :param data: loaded data 1316 :param model: the model to plot 1317 1292 1318 """ 1293 1319 x= numpy.linspace(start= qmin, … … 1332 1358 def _cal_chisqr(self, data=None, index=None): 1333 1359 """ 1334 1335 1360 Get handy Chisqr using the output from draw1D and 2D, 1361 instead of calling expansive CalcChisqr in guithread 1336 1362 """ 1337 1363 # default chisqr … … 1369 1395 1370 1396 1371 def profile(fn, *args, **kw):1372 import cProfile, pstats, os1373 global call_result1374 def call():1375 global call_result1376 call_result = fn(*args, **kw)1377 cProfile.runctx('call()', dict(call=call), {}, 'profile.out')1378 stats = pstats.Stats('profile.out')1379 #stats.sort_stats('time')1380 stats.sort_stats('calls')1381 stats.print_stats()1382 os.unlink('profile.out')1383 return call_result1397 #def profile(fn, *args, **kw): 1398 # import cProfile, pstats, os 1399 # global call_result 1400 # def call(): 1401 # global call_result 1402 # call_result = fn(*args, **kw) 1403 # cProfile.runctx('call()', dict(call=call), {}, 'profile.out') 1404 # stats = pstats.Stats('profile.out') 1405 # #stats.sort_stats('time') 1406 # stats.sort_stats('calls') 1407 # stats.print_stats() 1408 # os.unlink('profile.out') 1409 # return call_result 1384 1410 if __name__ == "__main__": 1385 1411 i = Plugin() -
sansview/perspectives/fitting/help_panel.py
r0df5d68 r5062bbf 1 1 #!/usr/bin/python 2 2 3 import wx 3 4 import wx.html as html 4 5 from wx.lib.splitter import MultiSplitterWindow 5 6 import os 6 def help():7 """8 Provide general online help text9 Future work: extend this function to allow topic selection10 """11 info_txt = "The inversion approach is based on Moore, J. Appl. Cryst. (1980) 13, 168-175.\n\n"12 info_txt += "P(r) is set to be equal to an expansion of base functions of the type "13 info_txt += "phi_n(r) = 2*r*sin(pi*n*r/D_max). The coefficient of each base functions "14 info_txt += "in the expansion is found by performing a least square fit with the "15 info_txt += "following fit function:\n\n"16 info_txt += "chi**2 = sum_i[ I_meas(q_i) - I_th(q_i) ]**2/error**2 + Reg_term\n\n"17 info_txt += "where I_meas(q) is the measured scattering intensity and I_th(q) is "18 info_txt += "the prediction from the Fourier transform of the P(r) expansion. "19 info_txt += "The Reg_term term is a regularization term set to the second derivative "20 info_txt += "d**2P(r)/dr**2 integrated over r. It is used to produce a smooth P(r) output.\n\n"21 info_txt += "The following are user inputs:\n\n"22 info_txt += " - Number of terms: the number of base functions in the P(r) expansion.\n\n"23 info_txt += " - Regularization constant: a multiplicative constant to set the size of "24 info_txt += "the regularization term.\n\n"25 info_txt += " - Maximum distance: the maximum distance between any two points in the system.\n"26 27 return info_txt28 29 class HelpDialog(wx.Dialog):30 def __init__(self, parent, id):31 32 wx.Dialog.__init__(self, parent, id, size=(400, 420))33 self.SetTitle("P(r) help")34 35 36 vbox = wx.BoxSizer(wx.VERTICAL)37 38 explanation = help()39 40 label_explain = wx.StaticText(self, -1, explanation, size=(350,320))41 42 vbox.Add(label_explain, 0, wx.ALL|wx.EXPAND, 15)43 7 44 8 45 static_line = wx.StaticLine(self, -1)46 vbox.Add(static_line, 0, wx.EXPAND, 0)47 48 button_OK = wx.Button(self, wx.ID_OK, "OK")49 50 sizer_button = wx.BoxSizer(wx.HORIZONTAL)51 sizer_button.Add((20, 20), 1, wx.EXPAND|wx.ADJUST_MINSIZE, 0)52 sizer_button.Add(button_OK, 0, wx.LEFT|wx.RIGHT|wx.ADJUST_MINSIZE, 10)53 vbox.Add(sizer_button, 0, wx.EXPAND|wx.BOTTOM|wx.TOP, 10)54 55 self.SetSizer(vbox)56 self.SetAutoLayout(True)57 58 self.Layout()59 self.Centre()60 61 9 class HelpWindow(wx.Frame): 10 """ 11 """ 62 12 def __init__(self, parent, id, title= 'HelpWindow', pageToOpen=None): 63 13 wx.Frame.__init__(self, parent, id, title, size=(850, 500)) 64 14 """ 65 15 contains help info 66 16 """ 67 17 … … 143 93 def OnButtonClicked(self, event): 144 94 """ 145 95 Function to diplay html page related to the hyperlinktext selected 146 96 """ 147 #link= "media/modelfunction.html"148 97 self.rhelp.LoadPage("media/modelfunction.html") 149 98 150 99 def OnLinkClicked(self, event): 151 100 """ 152 101 Function to diplay html page related to the hyperlinktext selected 153 102 """ 154 103 link= event.GetLinkInfo().GetHref() 155 104 self.rhelp.LoadPage(link) 105 """ 106 Example: :: 156 107 157 class ViewApp(wx.App): 158 def OnInit(self): 159 frame = HelpWindow(None, -1, 'HelpWindow') 160 frame.Show(True) 161 self.SetTopWindow(frame) 162 163 return True 164 108 class ViewApp(wx.App): 109 def OnInit(self): 110 frame = HelpWindow(None, -1, 'HelpWindow') 111 frame.Show(True) 112 self.SetTopWindow(frame) 113 114 return True 115 116 117 if __name__ == "__main__": 118 app = ViewApp(0) 119 app.MainLoop() 165 120 166 if __name__ == "__main__": 167 app = ViewApp(0) 168 app.MainLoop() 121 """ -
sansview/perspectives/fitting/hint_fitpage.py
r533d38d r5062bbf 1 1 2 import wx 2 3 … … 4 5 class HintFitPage(wx.ScrolledWindow): 5 6 """ 6 7 This class provide general structure of fitpanel page 7 8 """ 8 9 ## Internal name for the AUI manager … … 12 13 13 14 def __init__(self, parent): 15 """ 16 """ 14 17 wx.ScrolledWindow.__init__(self, parent, 15 18 style= wx.FULL_REPAINT_ON_RESIZE ) … … 21 24 def do_layout(self): 22 25 """ 23 26 Draw the page 24 27 """ 25 28 name="Hint" … … 41 44 42 45 def createMemento(self): 46 """ 47 """ 43 48 return 44 45 class HelpWindow(wx.Frame): 46 def __init__(self, parent, id, title): 47 wx.Frame.__init__(self, parent, id, title, size=(570, 400)) 48 49 self.page = HintFitPage(self) 50 self.Centre() 51 self.Show(True) 52 53 if __name__=="__main__": 54 app = wx.App() 55 HelpWindow(None, -1, 'HelpWindow') 56 app.MainLoop() 57 49 """ 50 Example: :: 51 52 class HelpWindow(wx.Frame): 53 def __init__(self, parent, id, title): 54 wx.Frame.__init__(self, parent, id, title, size=(570, 400)) 55 56 self.page = HintFitPage(self) 57 self.Centre() 58 self.Show(True) 59 60 if __name__=="__main__": 61 app = wx.App() 62 HelpWindow(None, -1, 'HelpWindow') 63 app.MainLoop() 64 """ -
sansview/perspectives/fitting/model_thread.py
rf72333f r5062bbf 1 2 1 3 import time 2 4 from data_util.calcthread import CalcThread … … 4 6 import numpy,math 5 7 from DataLoader.smearing_2d import Smearer2D 6 import fitpage 8 7 9 class Calc2D(CalcThread): 8 10 """ 9 Compute 2D model 10 This calculation assumes a 2-fold symmetry of the model 11 where points are computed for one half of the detector 12 and I(qx, qy) = I(-qx, -qy) is assumed. 13 """ 14 11 Compute 2D model 12 This calculation assumes a 2-fold symmetry of the model 13 where points are computed for one half of the detector 14 and I(qx, qy) = I(-qx, -qy) is assumed. 15 """ 15 16 def __init__(self, x, y, data,model,smearer,qmin, qmax,qstep, 16 17 completefn = None, … … 37 38 def compute(self): 38 39 """ 39 40 Compute the data given a model function 40 41 """ 41 42 self.starttime = time.time() … … 125 126 126 127 class Calc1D(CalcThread): 127 """Compute 1D data""" 128 128 """ 129 Compute 1D data 130 """ 129 131 def __init__(self, x, model, 130 132 data=None, … … 137 139 worktime = 0.01 138 140 ): 141 """ 142 """ 139 143 CalcThread.__init__(self,completefn, 140 144 updatefn, … … 151 155 def compute(self): 152 156 """ 153 Compute model 1d value given qmin , qmax , x value 154 """ 155 157 Compute model 1d value given qmin , qmax , x value 158 """ 156 159 self.starttime = time.time() 157 160 output = numpy.zeros((len(self.x))) … … 166 169 output[index] = self.model.evalDistribution(self.x[index]) 167 170 168 elapsed = time.time() -self.starttime171 elapsed = time.time() - self.starttime 169 172 170 self.complete(x= self.x[index], y=output[index],171 elapsed=elapsed,index=index, model= self.model, data=self.data)172 173 self.complete(x=self.x[index], y=output[index], 174 elapsed=elapsed,index=index, model=self.model, 175 data=self.data) 173 176 174 177 def results(self): 175 178 """ 176 179 Send resuts of the computation 177 180 """ 178 181 return [self.out, self.index] 179 180 class CalcCommandline: 181 def __init__(self, n=20000): 182 #print thread.get_ident() 183 from sans.models.CylinderModel import CylinderModel 184 185 model = CylinderModel() 186 182 183 """ 184 Example: :: 185 186 class CalcCommandline: 187 def __init__(self, n=20000): 188 #print thread.get_ident() 189 from sans.models.CylinderModel import CylinderModel 190 191 model = CylinderModel() 192 193 194 print model.runXY([0.01, 0.02]) 195 196 qmax = 0.01 197 qstep = 0.0001 198 self.done = False 199 200 x = numpy.arange(-qmax, qmax+qstep*0.01, qstep) 201 y = numpy.arange(-qmax, qmax+qstep*0.01, qstep) 202 203 204 calc_thread_2D = Calc2D(x, y, None, model.clone(),None, 205 -qmax, qmax,qstep, 206 completefn=self.complete, 207 updatefn=self.update , 208 yieldtime=0.0) 187 209 188 print model.runXY([0.01, 0.02]) 189 190 qmax = 0.01 191 qstep = 0.0001 192 self.done = False 193 194 x = numpy.arange(-qmax, qmax+qstep*0.01, qstep) 195 y = numpy.arange(-qmax, qmax+qstep*0.01, qstep) 210 calc_thread_2D.queue() 211 calc_thread_2D.ready(2.5) 212 213 while not self.done: 214 time.sleep(1) 196 215 216 def update(self,output): 217 print "update" 197 218 198 calc_thread_2D = Calc2D(x, y, None, model.clone(),-qmax, qmax,qstep, 199 completefn=self.complete, 200 updatefn=self.update , 201 yieldtime=0.0) 202 203 calc_thread_2D.queue() 204 calc_thread_2D.ready(2.5) 205 206 while not self.done: 207 time.sleep(1) 208 209 def update(self,output): 210 print "update" 211 212 def complete(self, image, data, model, elapsed, qmin, qmax, qstep ): 213 print "complete" 214 self.done = True 215 216 if __name__ == "__main__": 217 CalcCommandline() 218 219 def complete(self, image, data, model, elapsed, qmin, qmax,index, qstep ): 220 print "complete" 221 self.done = True 222 223 if __name__ == "__main__": 224 CalcCommandline() 225 """ -
sansview/perspectives/fitting/models.py
rce07fa8 r5062bbf 1 #TODO: add comments to document this module2 #TODO: clean-up the exception handling.3 4 #TODO: clean-up the FractalAbsModel and PowerLawAbsModel menu items. Those5 # model definitions do not belong here. They belong with the rest of the6 # models.7 1 8 2 import wx … … 23 17 24 18 def log(message): 19 """ 20 """ 25 21 out = open("plugins.log", 'a') 26 22 out.write("%10g: %s\n" % (time.clock(), message)) … … 28 24 29 25 def findModels(): 26 """ 27 """ 30 28 log("looking for models in: %s/plugins" % os.getcwd()) 31 29 if os.path.isdir('plugins'): … … 35 33 def _check_plugin(model, name): 36 34 """ 37 Do some checking before model adding plugins in the list 38 @param model: class model to add into the plugin list 39 @param name:name of the module plugin 40 @return model: model if valid model or None if not valid 35 Do some checking before model adding plugins in the list 36 37 :param model: class model to add into the plugin list 38 :param name:name of the module plugin 39 40 :return model: model if valid model or None if not valid 41 41 42 """ 42 43 #Check is the plugin is of type Model1DPlugin … … 74 75 75 76 def _findModels(dir): 77 """ 78 """ 76 79 # List of plugin objects 77 80 plugins = [] … … 114 117 class ModelList(object): 115 118 """ 116 119 Contains dictionary of model and their type 117 120 """ 118 121 def __init__(self): 119 self.mydict={} 122 """ 123 """ 124 self.mydict = {} 120 125 121 126 def set_list(self, name, mylist): 122 127 """ 123 @param name: the type of the list 124 @param mylist: the list to add 128 :param name: the type of the list 129 :param mylist: the list to add 130 125 131 """ 126 132 if name not in self.mydict.keys(): … … 130 136 def get_list(self): 131 137 """ 132 138 return all the list stored in a dictionary object 133 139 """ 134 140 return self.mydict 135 141 136 142 class ModelManager: 143 """ 144 """ 137 145 ## external dict for models 138 146 model_combobox = ModelList() … … 156 164 def _getModelList(self): 157 165 """ 158 List of models we want to make available by default 159 for this application 160 161 @return: the next free event ID following the new menu events 166 List of models we want to make available by default 167 for this application 168 169 :return: the next free event ID following the new menu events 170 162 171 """ 163 172 ## form factor … … 255 264 from sans.models.HayterMSAStructure import HayterMSAStructure 256 265 self.struct_list.append(HayterMSAStructure) 257 258 259 ##shape-independent models 260 266 267 ##shape-independent models 261 268 from sans.models.PowerLawAbsModel import PowerLawAbsModel 262 269 self.shape_indep_list.append( PowerLawAbsModel ) … … 293 300 self.shape_indep_list.append(PorodModel ) 294 301 295 296 302 #FractalModel (a c-model)will be used. 297 303 #from sans.models.FractalAbsModel import FractalAbsModel … … 303 309 from sans.models.LineModel import LineModel 304 310 self.shape_indep_list.append(LineModel) 305 306 311 307 312 #Looking for plugins 308 313 self.plugins = findModels() 309 310 314 return 0 311 315 … … 313 317 def populate_menu(self, modelmenu, event_owner): 314 318 """ 315 Populate a menu with our models 316 317 @param id: first menu event ID to use when binding the menu events 318 @param modelmenu: wx.Menu object to populate 319 @param event_owner: wx object to bind the menu events to 320 @return: the next free event ID following the new menu events 319 Populate a menu with our models 320 321 :param id: first menu event ID to use when binding the menu events 322 :param modelmenu: wx.Menu object to populate 323 :param event_owner: wx object to bind the menu events to 324 325 :return: the next free event ID following the new menu events 326 321 327 """ 322 328 ## Fill model lists … … 327 333 self.event_owner = event_owner 328 334 329 330 335 shape_submenu = wx.Menu() 331 336 shape_indep_submenu = wx.Menu() … … 334 339 multip_models = wx.Menu() 335 340 ## create menu with shape 336 self._fill_simple_menu( menuinfo =["Shapes",shape_submenu," simple shape"],337 list1 = self.shape_list)338 339 self._fill_simple_menu( menuinfo =["Shape-Independent",shape_indep_submenu,341 self._fill_simple_menu(menuinfo=["Shapes",shape_submenu," simple shape"], 342 list1=self.shape_list) 343 344 self._fill_simple_menu(menuinfo=["Shape-Independent",shape_indep_submenu, 340 345 "List of shape-independent models"], 341 list1 =self.shape_indep_list )342 343 self._fill_simple_menu( menuinfo=["Structure Factors",structure_factor,346 list1=self.shape_indep_list ) 347 348 self._fill_simple_menu(menuinfo=["Structure Factors",structure_factor, 344 349 "List of Structure factors models" ], 345 list1= self.struct_list)346 347 self._fill_plugin_menu( menuinfo =["Customized Models", added_models,350 list1=self.struct_list) 351 352 self._fill_plugin_menu(menuinfo=["Customized Models", added_models, 348 353 "List of additional models"], 349 list1= self.plugins)354 list1=self.plugins) 350 355 351 356 self._fill_menu(menuinfo=["P(Q)*S(Q)",multip_models, 352 357 "mulplication of 2 models"], 353 list1 = self.multiplication_factor , 354 list2 = self.struct_list) 355 356 358 list1=self.multiplication_factor , 359 list2= self.struct_list) 357 360 return 0 358 361 359 def _fill_plugin_menu(self, menuinfo, list1):360 """ 361 362 def _fill_plugin_menu(self, menuinfo, list1): 363 """ 364 fill the plugin menu with costumized models 362 365 """ 363 366 if len(list1)==0: … … 367 370 self._fill_simple_menu( menuinfo,list1) 368 371 369 370 def _fill_simple_menu(self,menuinfo, list1): 371 """ 372 Fill the menu with list item 373 @param modelmenu: the menu to fill 374 @param menuinfo: submenu item for the first column of this modelmenu 375 with info.Should be a list : 376 [name(string) , menu(wx.menu), help(string)] 377 @param list1: contains item (form factor )to fill modelmenu second column 372 def _fill_simple_menu(self, menuinfo, list1): 373 """ 374 Fill the menu with list item 375 376 :param modelmenu: the menu to fill 377 :param menuinfo: submenu item for the first column of this modelmenu 378 with info.Should be a list : 379 [name(string) , menu(wx.menu), help(string)] 380 :param list1: contains item (form factor )to fill modelmenu second column 381 378 382 """ 379 383 if len(list1)>0: … … 399 403 self.modelmenu.AppendMenu(id, menuinfo[0],menuinfo[1],menuinfo[2]) 400 404 401 402 403 def _fill_menu(self,menuinfo, list1,list2 ): 404 """ 405 Fill the menu with list item 406 @param menuinfo: submenu item for the first column of this modelmenu 407 with info.Should be a list : 408 [name(string) , menu(wx.menu), help(string)] 409 @param list1: contains item (form factor )to fill modelmenu second column 410 @param list2: contains item (Structure factor )to fill modelmenu third column 405 def _fill_menu(self, menuinfo, list1, list2): 406 """ 407 Fill the menu with list item 408 409 :param menuinfo: submenu item for the first column of this modelmenu 410 with info.Should be a list : 411 [name(string) , menu(wx.menu), help(string)] 412 :param list1: contains item (form factor )to fill modelmenu second column 413 :param list2: contains item (Structure factor )to fill modelmenu 414 third column 415 411 416 """ 412 417 if len(list1)>0: … … 437 442 self.modelmenu.AppendMenu(id,menuinfo[0],menuinfo[1], menuinfo[2]) 438 443 439 440 441 442 444 def _on_model(self, evt): 443 445 """ 444 React to a model menu event 445 @param event: wx menu event 446 React to a model menu event 447 448 :param event: wx menu event 449 446 450 """ 447 451 if int(evt.GetId()) in self.form_factor_dict.keys(): 448 452 from sans.models.MultiplicationModel import MultiplicationModel 449 453 model1, model2 = self.form_factor_dict[int(evt.GetId())] 450 model = MultiplicationModel(model1, model2) 451 454 model = MultiplicationModel(model1, model2) 452 455 else: 453 456 model= self.struct_factor_dict[str(evt.GetId())]() 454 455 457 evt = ModelEvent( model= model ) 456 458 wx.PostEvent(self.event_owner, evt) 457 459 458 460 def get_model_list(self): 459 """ @ return dictionary of models for fitpanel use """ 461 """ 462 return dictionary of models for fitpanel use 463 464 """ 460 465 self.model_combobox.set_list("multiplication", self.multiplication_factor) 461 466 return self.model_combobox -
sansview/perspectives/fitting/pagestate.py
r35b556d r5062bbf 1 """ 2 This software was developed by the University of Tennessee as part of the 3 Distributed Data Analysis of Neutron Scattering Experiments (DANSE) 4 project funded by the US National Science Foundation. 5 6 See the license text in license.txt 7 8 copyright 2009, University of Tennessee 9 """ 1 2 3 ################################################################################ 4 #This software was developed by the University of Tennessee as part of the 5 #Distributed Data Analysis of Neutron Scattering Experiments (DANSE) 6 #project funded by the US National Science Foundation. 7 # 8 #See the license text in license.txt 9 # 10 #copyright 2009, University of Tennessee 11 ################################################################################ 12 10 13 import time 11 14 import os … … 87 90 class PageState(object): 88 91 """ 89 92 Contains information to reconstruct a page of the fitpanel. 90 93 """ 91 94 def __init__(self, parent=None, model=None, data=None): 92 95 93 96 """ 94 Initialize the current state 95 @param model: a selected model within a page 96 @param data: 97 Initialize the current state 98 99 :param model: a selected model within a page 100 :param data: 101 97 102 """ 98 103 self.file = None … … 201 206 def clone(self): 202 207 """ 203 208 Create a new copy of the current object 204 209 """ 205 210 model = None … … 271 276 def _repr_helper(self, list, rep): 272 277 """ 273 278 Helper method to print a state 274 279 """ 275 280 for item in list: … … 288 293 def __repr__(self): 289 294 """ 290 295 output string for printing 291 296 """ 292 297 rep = "\nState name: %s\n"%self.file … … 321 326 def _toXML_helper(self, list, element, newdoc): 322 327 """ 323 328 Helper method to create xml file for saving state 324 329 """ 325 330 for item in list: … … 339 344 def toXML(self, file="fitting_state.fitv", doc=None, entry_node=None): 340 345 """ 341 Writes the state of the InversionControl panel to file, as XML. 342 343 Compatible with standalone writing, or appending to an 344 already existing XML document. In that case, the XML document 345 is required. An optional entry node in the XML document may also be given. 346 347 @param file: file to write to 348 @param doc: XML document object [optional] 349 @param entry_node: XML node within the XML document at which we will append the data [optional] 346 Writes the state of the InversionControl panel to file, as XML. 347 348 Compatible with standalone writing, or appending to an 349 already existing XML document. In that case, the XML document 350 is required. An optional entry node in the XML document may also be given. 351 352 :param file: file to write to 353 :param doc: XML document object [optional] 354 :param entry_node: XML node within the XML document at which we will append the data [optional] 355 350 356 """ 351 357 from xml.dom.minidom import getDOMImplementation … … 430 436 def _fromXML_helper(self, node, list): 431 437 """ 432 438 Helper function to write state to xml 433 439 """ 434 440 for item in node: … … 448 454 def fromXML(self, file=None, node=None): 449 455 """ 450 Load fitting state from a file 451 452 @param file: .fitv file 453 @param node: node of a XML document to read from 456 Load fitting state from a file 457 458 :param file: .fitv file 459 :param node: node of a XML document to read from 460 454 461 """ 455 462 … … 512 519 class Reader(CansasReader): 513 520 """ 514 521 Class to load a .fitv fitting file 515 522 """ 516 523 ## File type … … 524 531 def __init__(self, call_back=None, cansas=True): 525 532 """ 526 Initialize the call-back method to be called 527 after we load a file 528 @param call_back: call-back method 529 @param cansas: True = files will be written/read in CanSAS format 530 False = write CanSAS format 533 Initialize the call-back method to be called 534 after we load a file 535 536 :param call_back: call-back method 537 :param cansas: True = files will be written/read in CanSAS format 538 False = write CanSAS format 531 539 532 540 """ … … 538 546 def read(self, path): 539 547 """ 540 541 542 @param path: file path543 @return: None548 Load a new P(r) inversion state from file 549 550 :param path: file path 551 544 552 """ 545 553 if self.cansas == True: … … 548 556 def _data2d_to_xml_doc(self, datainfo): 549 557 """ 550 Create an XML document to contain the content of a Data2D 551 552 @param datainfo: Data2D object 558 Create an XML document to contain the content of a Data2D 559 560 :param datainfo: Data2D object 561 553 562 """ 554 563 if not issubclass(datainfo.__class__, Data2D): … … 722 731 for note in item.notes: 723 732 write_node(doc, node, "SASprocessnote", note) 724 725 733 # Return the document, and the SASentry node associated with 726 734 # the data we just wrote 727 735 return doc, entry_node 728 736 729 730 737 def _parse_state(self, entry): 731 738 """ 732 Read a fit result from an XML node 733 @param entry: XML node to read from 734 @return: PageState object 739 Read a fit result from an XML node 740 741 :param entry: XML node to read from 742 743 :return: PageState object 735 744 """ 736 745 # Create an empty state … … 747 756 def _parse_entry_2d(self, dom): 748 757 """ 749 Parse a SASentry 750 751 @param node: SASentry node 752 @return: Data2D object 753 """ 754 758 Parse a SASentry 759 760 :param node: SASentry node 761 762 :return: Data2D object 763 764 """ 755 765 data_info = Data2D() 756 766 … … 1083 1093 def _read_cansas(self, path): 1084 1094 """ 1085 Load data and P(r) information from a CanSAS XML file. 1086 1087 @param path: file path 1088 @return: Data1D object if a single SASentry was found, 1089 or a list of Data1D objects if multiple entries were found, 1090 or None of nothing was found 1091 @raise RuntimeError: when the file can't be opened 1092 @raise ValueError: when the length of the data vectors are inconsistent 1095 Load data and P(r) information from a CanSAS XML file. 1096 1097 :param path: file path 1098 1099 :return: Data1D object if a single SASentry was found, 1100 or a list of Data1D objects if multiple entries were found, 1101 or None of nothing was found 1102 1103 :raise RuntimeError: when the file can't be opened 1104 :raise ValueError: when the length of the data vectors are inconsistent 1105 1093 1106 """ 1094 1107 output = [] … … 1155 1168 def write(self, filename, datainfo=None, fitstate=None): 1156 1169 """ 1157 Write the content of a Data1D as a CanSAS XML file 1158 1159 @param filename: name of the file to write 1160 @param datainfo: Data1D object 1161 @param fitstate: PageState object 1170 Write the content of a Data1D as a CanSAS XML file 1171 1172 :param filename: name of the file to write 1173 :param datainfo: Data1D object 1174 :param fitstate: PageState object 1175 1162 1176 """ 1163 1177 # Sanity check … … 1182 1196 fitstate.toXML(file=filename) 1183 1197 1184 1185 if __name__ == "__main__": 1186 state = PageState(parent=None) 1187 state.toXML() 1188 print "state", state 1198 """ 1199 Example: :: 1200 1201 if __name__ == "__main__": 1202 state = PageState(parent=None) 1203 state.toXML() 1204 print "state", state 1205 1206 """ -
sansview/perspectives/fitting/simfitpage.py
r6e9976b r5062bbf 14 14 def get_fittableParam( model): 15 15 """ 16 @return list of fittable parameters name of a model 17 @param model: the model used 16 return list of fittable parameters name of a model 17 18 :param model: the model used 19 18 20 """ 19 21 fittable_param=[] … … 30 32 class SimultaneousFitPage(wx.ScrolledWindow): 31 33 """ 32 33 34 34 Simultaneous fitting panel 35 All that needs to be defined are the 36 two data members window_name and window_caption 35 37 """ 36 38 ## Internal name for the AUI manager … … 43 45 wx.ScrolledWindow.__init__(self, parent,style= wx.FULL_REPAINT_ON_RESIZE ) 44 46 """ 45 47 Simultaneous page display 46 48 """ 47 49 ##Font size … … 67 69 self.set_layout() 68 70 69 70 71 71 def define_page_structure(self): 72 72 """ 73 73 Create empty sizer for a panel 74 74 """ 75 75 self.vbox = wx.BoxSizer(wx.VERTICAL) … … 86 86 87 87 def set_scroll(self): 88 """ 89 """ 88 90 self.SetScrollbars(20,20,25,65) 89 91 self.Layout() … … 91 93 def set_layout(self): 92 94 """ 93 95 layout 94 96 """ 95 97 self.vbox.Layout() 96 98 self.vbox.Fit(self) 97 99 self.SetSizer(self.vbox) 98 99 100 self.set_scroll() 100 101 self.Centre() … … 102 103 def onRemove(self, event): 103 104 """ 104 105 Remove constraint fields 105 106 """ 106 107 if len(self.constraints_list)==1: … … 123 124 break 124 125 125 126 126 def onFit(self,event): 127 """ signal for fitting""" 127 """ 128 signal for fitting 129 130 """ 128 131 ## making sure all parameters content a constraint 129 132 ## validity of the constraint expression is own by fit engine … … 145 148 wx.PostEvent(self.parent.Parent, StatusEvent(status= msg )) 146 149 147 148 149 150 def set_manager(self, manager): 150 151 """ 151 set panel manager 152 @param manager: instance of plugin fitting 152 set panel manager 153 154 :param manager: instance of plugin fitting 155 153 156 """ 154 157 self.manager = manager 155 158 156 157 159 def check_all_model_name(self,event): 158 160 """ 159 161 check all models names 160 162 """ 161 163 self.model_toFit=[] … … 179 181 self._hide_constraint() 180 182 181 182 183 def check_model_name(self,event): 183 184 """ 184 185 Save information related to checkbox and their states 185 186 """ 186 187 self.model_toFit=[] … … 208 209 else: 209 210 self.cb1.SetValue(False) 210 211 212 211 213 212 def draw_page(self): 214 213 """ 215 214 Draw a sizer containing couples of data and model 216 215 """ 217 218 216 self.model_list=[] 219 217 self.model_toFit=[] … … 227 225 self.manager.schedule_for_fit( value=0,fitproblem =item[1]) 228 226 229 self.sizer1.Clear(True) 230 231 227 self.sizer1.Clear(True) 232 228 box_description= wx.StaticBox(self, -1,"Fit Combinations") 233 229 boxsizer1 = wx.StaticBoxSizer(box_description, wx.VERTICAL) 234 230 sizer_title = wx.BoxSizer(wx.HORIZONTAL) 235 231 sizer_couples = wx.GridBagSizer(5,5) 236 237 232 #------------------------------------------------------ 238 233 if len(self.page_finder)==0: … … 257 252 ## draw fit button 258 253 self._fill_sizer_fit() 259 260 254 #-------------------------------------------------------- 261 255 boxsizer1.Add(sizer_title, flag= wx.TOP|wx.BOTTOM,border=5) … … 269 263 def _store_model(self): 270 264 """ 271 265 Store selected model 272 266 """ 273 267 if len(self.model_toFit) < 2: … … 278 272 self.constraint_dict[page] = model 279 273 280 281 274 def _display_constraint(self, event): 282 275 """ 283 276 Show fields to add constraint 284 277 """ 285 278 if len(self.model_toFit)< 2: … … 296 289 return 297 290 298 299 291 def _show_constraint(self): 300 292 """ 301 293 Show constraint fields 302 294 """ 303 295 self.btAdd.Show(True) … … 340 332 btRemove.SetToolTipString("Remove constraint.") 341 333 342 343 334 for page,model in self.constraint_dict.iteritems(): 344 335 ## check if all parameters have been selected for constraint … … 350 341 351 342 wx.EVT_COMBOBOX(model_cbox,-1, self._on_select_model) 352 353 343 354 344 sizer_constraint.Add(model_cbox, flag= wx.RIGHT|wx.EXPAND,border=10) 355 345 sizer_constraint.Add(param_cbox, flag= wx.RIGHT|wx.EXPAND,border=5) … … 363 353 ##[combobox1, combobox2,=,textcrtl, remove button ] 364 354 self.constraints_list.append([model_cbox, param_cbox, egal_txt, ctl2,btRemove,sizer_constraint]) 365 366 355 367 356 self.nb_constraint += 1 368 357 self.sizer_constraints.Layout() … … 372 361 def _hide_constraint(self): 373 362 """ 374 363 hide buttons related constraint 375 364 """ 376 365 for page in self.page_finder.iterkeys(): … … 382 371 self.btAdd.Hide() 383 372 self._store_model() 384 385 373 self.constraints_list=[] 386 374 self.sizer_constraints.Clear(True) … … 389 377 self.SetScrollbars(20,20,25,65) 390 378 self.AdjustScrollbars() 391 392 393 379 394 380 def _on_select_model(self, event): 395 381 """ 396 382 fill combox box with list of parameters 397 383 """ 398 384 ##This way PC/MAC both work, instead of using event.GetClientData(). … … 404 390 if length < 1: 405 391 return 406 407 392 param_cbox = self.constraints_list[length-1][1] 408 393 param_cbox.Clear() … … 412 397 413 398 param_cbox.Show(True) 414 415 416 399 self.sizer2.Layout() 417 400 self.SetScrollbars(20,20,25,65) 418 401 419 420 402 def _on_select_param(self, event): 421 403 """ 422 404 Store the appropriate constraint in the page_finder 423 405 """ 424 406 ##This way PC/MAC both work, instead of using event.GetClientData(). … … 435 417 ctl2 = self.constraints_list[length-1][3] 436 418 ctl2.Show(True) 437 438 439 419 #self.sizer2.Layout() 440 420 self.SetScrollbars(20,20,25,65) 441 421 442 443 422 def _onAdd_constraint(self, event): 444 423 """ 445 Add another line for constraint 446 """ 447 424 Add another line for constraint 425 """ 448 426 if not self.show_constraint.GetValue(): 449 427 msg= " Select Yes to add Constraint " 450 428 wx.PostEvent(self.parent.Parent, StatusEvent(status= msg )) 451 429 return 452 453 430 ## check that a constraint is added before allow to add another cosntraint 454 455 431 for item in self.constraints_list: 456 432 model_cbox = item[0] … … 470 446 wx.PostEvent(self.parent.Parent, StatusEvent(status= msg )) 471 447 return 472 473 448 ## some model or parameters can be constrained 474 449 self._show_constraint() … … 476 451 def _fill_sizer_fit(self): 477 452 """ 478 Draw fitfit button453 Draw fit button 479 454 """ 480 455 self.sizer3.Clear(True) … … 501 476 def _fill_sizer_constraint(self): 502 477 """ 503 504 """ 505 msg = "Select at least 2 model to add constraint "478 Fill sizer containing constraint info 479 """ 480 msg = "Select at least 2 model to add constraint " 506 481 wx.PostEvent(self.parent.Parent, StatusEvent(status= msg )) 507 482 508 483 self.sizer2.Clear(True) 509 510 484 box_description= wx.StaticBox(self, -1,"Fit Constraints") 511 485 boxsizer1 = wx.StaticBoxSizer(box_description, wx.VERTICAL) … … 516 490 self.hide_constraint = wx.RadioButton(self, -1, 'No', (10, 10), style=wx.RB_GROUP) 517 491 self.show_constraint = wx.RadioButton(self, -1, 'Yes', (10, 30)) 518 519 492 self.Bind( wx.EVT_RADIOBUTTON, self._display_constraint, 520 493 id= self.hide_constraint.GetId() ) 521 522 494 self.Bind( wx.EVT_RADIOBUTTON, self._display_constraint, 523 495 id= self.show_constraint.GetId() ) … … 538 510 self.btAdd.Hide() 539 511 540 541 512 text_hint = wx.StaticText(self,-1,"Example: [M0][paramter] = M1.parameter") 542 513 sizer_button.Add(text_hint, 0 , wx.LEFT|wx.EXPAND|wx.ADJUST_MINSIZE, 10) 543 514 sizer_button.Add(self.btAdd, 0, wx.LEFT|wx.EXPAND|wx.ADJUST_MINSIZE, 10) 544 515 545 546 516 boxsizer1.Add(sizer_title, flag= wx.TOP|wx.BOTTOM,border=10) 547 517 boxsizer1.Add(self.sizer_constraints, flag= wx.TOP|wx.BOTTOM,border=10) … … 551 521 self.sizer2.Layout() 552 522 self.SetScrollbars(20,20,25,65) 553 554 555 523 556 524 def _set_constraint(self): 557 525 """ 558 559 560 561 526 get values from the constrainst textcrtl ,parses them into model name 527 parameter name and parameters values. 528 store them in a list self.params .when when params is not empty set_model 529 uses it to reset the appropriate model and its appropriates parameters 562 530 """ 563 531 for item in self.constraints_list: 564 565 532 model = item[0].GetClientData(item[0].GetCurrentSelection()) 566 533 param = item[1].GetString(item[1].GetCurrentSelection()) … … 580 547 self.page_finder[page].set_model_param(param,constraint) 581 548 break 582 583 584 549 585 550 def _fill_sizer_model_list(self,sizer): 586 551 """ 587 Receive a dictionary containing information to display model name 588 @param page_finder: the dictionary containing models information 552 Receive a dictionary containing information to display model name 553 554 :param page_finder: the dictionary containing models information 555 589 556 """ 590 557 ix = 0 … … 597 564 sizer.Add(new_name,(iy, ix),(1,1), 598 565 wx.LEFT|wx.EXPAND|wx.ADJUST_MINSIZE, 15) 599 600 ix +=2 566 ix += 2 601 567 model_type = wx.StaticText(self, -1, ' Model Type') 602 568 model_type.SetBackgroundColour('grey') 603 569 sizer.Add(model_type,(iy, ix),(1,1), 604 570 wx.EXPAND|wx.ADJUST_MINSIZE, 0) 605 ix += 1571 ix += 1 606 572 data_used = wx.StaticText(self, -1, ' Used Data') 607 573 data_used.SetBackgroundColour('grey') … … 619 585 wx.EVT_CHECKBOX(self, cb.GetId(), self.check_model_name) 620 586 621 ix += 2587 ix += 2 622 588 type = model.__class__.__name__ 623 589 model_type = wx.StaticText(self, -1, str(type)) 624 590 sizer.Add(model_type,( iy,ix),(1,1), wx.EXPAND|wx.ADJUST_MINSIZE, 0) 625 591 626 ix += 1592 ix += 1 627 593 data = value.get_fit_data() 628 594 data_used= wx.StaticText(self, -1, str(data.name)) … … 633 599 except: 634 600 pass 635 iy += 1601 iy += 1 636 602 sizer.Add((20,20),( iy,ix),(1,1), wx.LEFT|wx.EXPAND|wx.ADJUST_MINSIZE, 15) 637 603 sizer.Layout() 638 639 640 641 642 class HelpWindow(wx.Frame):643 def __init__(self, parent, id, title):644 wx.Frame.__init__(self, parent, id, title, size=(570, 400))645 646 page_finder ={}647 ## create random data648 from danse.common.plottools.plottables import Data1D649 data= Data1D(x=[1,2], y=[3,4], dy=[0.1, 0,1])650 data.name="mydata.txt"651 ## create model652 from sans.models.CylinderModel import CylinderModel653 model = CylinderModel()654 model.name="M0"655 656 657 from fitproblem import FitProblem658 page_finder["page"]= FitProblem()659 ## fill the page_finder660 page_finder["page"].add_fit_data(data)661 page_finder["page"].set_model(model)662 self.page = SimultaneousFitPage(self, page_finder=page_finder)663 664 665 666 self.Centre()667 self.Show(True)668 669 670 671 if __name__=="__main__":672 app = wx.App()673 HelpWindow(None, -1, 'HelpWindow')674 app.MainLoop()675
Note: See TracChangeset
for help on using the changeset viewer.