Source code for sas.perspectives.fitting.simfitpage

"""
    Simultaneous fit page
"""
import sys
from collections import namedtuple

import wx
import wx.lib.newevent
from wx.lib.scrolledpanel import ScrolledPanel

from sas.guiframe.events import StatusEvent
from sas.guiframe.panel_base import PanelBase
from sas.guiframe.events import PanelOnFocusEvent
from sas.guiframe.utils import IdList
from sas.guiframe.documentation_window import DocumentationWindow

#Control panel width 
if sys.platform.count("darwin") == 0:
    PANEL_WID = 420
    FONT_VARIANT = 0
else:
    PANEL_WID = 490
    FONT_VARIANT = 1


# Each constraint requires five widgets and sizer.  Package them in
# a named tuple for easy access.
ConstraintLine = namedtuple('ConstraintLine',
        'model_cbox param_cbox egal_txt constraint btRemove sizer')

[docs]def get_fittableParam(model): """ return list of fittable parameters from a model :param model: the model used """ fittable_param = [] for item in model.getParamList(): if not item in model.getDispParamList(): if not item in model.non_fittable: fittable_param.append(item) for item in model.fixed: fittable_param.append(item) return fittable_param
[docs]class SimultaneousFitPage(ScrolledPanel, PanelBase): """ Simultaneous fitting panel All that needs to be defined are the two data members window_name and window_caption """ ## Internal name for the AUI manager window_name = "simultaneous Fit page" ## Title to appear on top of the window window_caption = "Simultaneous Fit Page" ID_DOC = wx.NewId() ID_SET_ALL = wx.NewId() ID_FIT = wx.NewId() ID_ADD = wx.NewId() _id_pool = IdList() def __init__(self, parent, page_finder={}, id=wx.ID_ANY, batch_on=False, *args, **kwargs): ScrolledPanel.__init__(self, parent, id=id, style=wx.FULL_REPAINT_ON_RESIZE, *args, **kwargs) PanelBase.__init__(self, parent) """ Simultaneous page display """ self._ids = iter(self._id_pool) self.SetupScrolling() ##Font size self.SetWindowVariant(variant=FONT_VARIANT) self.uid = wx.NewId() self.parent = parent self.batch_on = batch_on ## store page_finder self.page_finder = page_finder ## list containing info to set constraint ## look like self.constraint_dict[page_id]= page self.constraint_dict = {} ## item list ## self.constraints_list=[combobox1, combobox2,=,textcrtl, button ] self.constraints_list = [] ## list of current model self.model_list = [] ## selected mdoel to fit self.model_toFit = [] ## number of constraint self.nb_constraint = 0 self.model_cbox_left = None self.model_cbox_right = None ## draw page self.define_page_structure() self.draw_page() self._set_save_flag(False)
[docs] def define_page_structure(self): """ Create empty sizers, their hierarchy and set the sizer for the panel """ self.vbox = wx.BoxSizer(wx.VERTICAL) self.sizer1 = wx.BoxSizer(wx.VERTICAL) self.sizer2 = wx.BoxSizer(wx.VERTICAL) self.sizer3 = wx.BoxSizer(wx.VERTICAL) self.sizer1.SetMinSize((PANEL_WID, -1)) self.sizer2.SetMinSize((PANEL_WID, -1)) self.sizer3.SetMinSize((PANEL_WID, -1)) self.vbox.Add(self.sizer1) self.vbox.Add(self.sizer2) self.vbox.Add(self.sizer3) self.SetSizer(self.vbox) self.Centre()
[docs] def draw_page(self): """ Construct the Simultaneous/Constrained fit page. fills the first region (sizer1) with the list of available fit page pairs of data and models. Then fills sizer2 with the checkbox for adding constraints, and finally fills sizer3 with the fit button and instructions. """ # create blank list of constraints self.model_list = [] self.model_toFit = [] self.constraints_list = [] self.constraint_dict = {} self.nb_constraint = 0 self.model_cbox_left = None self.model_cbox_right = None if len(self.model_list) > 0: for item in self.model_list: item[0].SetValue(False) self.manager.schedule_for_fit(value=0, uid=item[2]) #------------------------------------------------------- ## setup sizer1 (which fitpages to include) self.sizer1.Clear(True) box_description = wx.StaticBox(self, wx.ID_ANY, "Fit Combinations") boxsizer1 = wx.StaticBoxSizer(box_description, wx.VERTICAL) sizer_title = wx.BoxSizer(wx.HORIZONTAL) sizer_couples = wx.GridBagSizer(5, 5) #This if statement should be obsolete and can be removed in version 4 #Leave it here for now as no time to thoroughly test. However if no #fit page is found the menu item that calls this page is inactive # Nov. 22 2015 --PDB if len(self.page_finder) == 0: msg = " No fit combinations are found! \n\n" msg += " Please load data and set up " msg += "at least one fit panels first..." sizer_title.Add(wx.StaticText(self, wx.ID_ANY, msg)) else: ## store model self._store_model() self.cb1 = wx.CheckBox(self, wx.ID_ANY, 'Select all') self.cb1.SetValue(False) wx.EVT_CHECKBOX(self, self.cb1.GetId(), self.check_all_model_name) sizer_title.Add((10, 10), 0, wx.TOP | wx.BOTTOM | wx.EXPAND | wx.ADJUST_MINSIZE, border=5) sizer_title.Add(self.cb1, 0, wx.TOP | wx.BOTTOM | wx.LEFT | wx.EXPAND | wx.ADJUST_MINSIZE, border=5) ## draw list of model and data names self._fill_sizer_model_list(sizer_couples) boxsizer1.Add(sizer_title, flag=wx.TOP | wx.BOTTOM, border=5) boxsizer1.Add(sizer_couples, 1, flag=wx.TOP | wx.BOTTOM, border=5) self.sizer1.Add(boxsizer1, 1, wx.EXPAND | wx.ALL, 10) # self.sizer1.Layout() #-------------------------------------------------------- ## set up the other 2 sizers: the constraints list and the ## buttons (fit, help etc) sizer at the bottom of the page. ## Note: the if statement should be removed along with the above ## if statement as soon as it can be properly tested. ## Nov. 22 2015 --PDB if len(self.page_finder) > 0: ## draw the sizer containing constraint info if not self.batch_on: self._fill_sizer_constraint() ## draw fit button sizer self._fill_sizer_fit()
def _fill_sizer_model_list(self, sizer): """ Receive a dictionary containing information to display model name """ ix = 0 iy = 0 list = [] sizer.Clear(True) new_name = wx.StaticText(self, wx.ID_ANY, ' Model Title ', style=wx.ALIGN_CENTER) new_name.SetBackgroundColour('orange') new_name.SetForegroundColour(wx.WHITE) sizer.Add(new_name, (iy, ix), (1, 1), wx.LEFT | wx.EXPAND | wx.ADJUST_MINSIZE, 15) ix += 2 model_type = wx.StaticText(self, wx.ID_ANY, ' Model ') model_type.SetBackgroundColour('grey') model_type.SetForegroundColour(wx.WHITE) sizer.Add(model_type, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) ix += 1 data_used = wx.StaticText(self, wx.ID_ANY, ' Data ') data_used.SetBackgroundColour('grey') data_used.SetForegroundColour(wx.WHITE) sizer.Add(data_used, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) ix += 1 tab_used = wx.StaticText(self, wx.ID_ANY, ' FitPage ') tab_used.SetBackgroundColour('grey') tab_used.SetForegroundColour(wx.WHITE) sizer.Add(tab_used, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) for id, value in self.page_finder.iteritems(): if id not in self.parent.opened_pages: continue if self.batch_on != self.parent.get_page_by_id(id).batch_on: continue data_list = [] model_list = [] # get data name and model objetta for fitproblem in value.get_fit_problem(): data = fitproblem.get_fit_data() if not data.is_data: continue name = '-' if data is not None and data.is_data: name = str(data.name) data_list.append(name) model = fitproblem.get_model() if model is None: continue model_list.append(model) if len(model_list) == 0: continue # Draw sizer ix = 0 iy += 1 model = model_list[0] name = '_' if model is not None: name = str(model.name) cb = wx.CheckBox(self, wx.ID_ANY, name) cb.SetValue(False) cb.Enable(model is not None and data.is_data) sizer.Add(cb, (iy, ix), (1, 1), wx.LEFT | wx.EXPAND | wx.ADJUST_MINSIZE, 15) wx.EVT_CHECKBOX(self, cb.GetId(), self.check_model_name) ix += 2 model_type = wx.StaticText(self, wx.ID_ANY, model.__class__.__name__) sizer.Add(model_type, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) if self.batch_on: data_used = wx.ComboBox(self, wx.ID_ANY, style=wx.CB_READONLY) data_used.AppendItems(data_list) data_used.SetSelection(0) else: data_used = wx.StaticText(self, wx.ID_ANY, data_list[0]) ix += 1 sizer.Add(data_used, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) ix += 1 caption = value.get_fit_tab_caption() tab_caption_used = wx.StaticText(self, wx.ID_ANY, str(caption)) sizer.Add(tab_caption_used, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) self.model_list.append([cb, value, id, model]) iy += 1 sizer.Add((20, 20), (iy, ix), (1, 1), wx.LEFT | wx.EXPAND | wx.ADJUST_MINSIZE, 15) def _fill_sizer_constraint(self): """ Fill sizer containing constraint info """ msg = "Select at least 1 model to add constraint " wx.PostEvent(self.parent.parent, StatusEvent(status=msg)) self.sizer2.Clear(True) if self.batch_on: if self.sizer2.IsShown(): self.sizer2.Show(False) return box_description = wx.StaticBox(self, wx.ID_ANY, "Fit Constraints") boxsizer1 = wx.StaticBoxSizer(box_description, wx.VERTICAL) sizer_title = wx.BoxSizer(wx.HORIZONTAL) self.sizer_all_constraints = wx.BoxSizer(wx.HORIZONTAL) self.sizer_constraints = wx.BoxSizer(wx.VERTICAL) sizer_button = wx.BoxSizer(wx.HORIZONTAL) self.hide_constraint = wx.RadioButton(self, wx.ID_ANY, 'No', (10, 10), style=wx.RB_GROUP) self.show_constraint = wx.RadioButton(self, wx.ID_ANY, 'Yes', (10, 30)) self.Bind(wx.EVT_RADIOBUTTON, self._display_constraint, id=self.hide_constraint.GetId()) self.Bind(wx.EVT_RADIOBUTTON, self._display_constraint, id=self.show_constraint.GetId()) if self.batch_on: self.hide_constraint.Enable(False) self.show_constraint.Enable(False) self.hide_constraint.SetValue(True) self.show_constraint.SetValue(False) sizer_title.Add(wx.StaticText(self, wx.ID_ANY, " Model")) sizer_title.Add((10, 10)) sizer_title.Add(wx.StaticText(self, wx.ID_ANY, " Parameter")) sizer_title.Add((10, 10)) sizer_title.Add(wx.StaticText(self, wx.ID_ANY, " Add Constraint?")) sizer_title.Add((10, 10)) sizer_title.Add(self.show_constraint) sizer_title.Add(self.hide_constraint) sizer_title.Add((10, 10)) self.btAdd = wx.Button(self, self.ID_ADD, 'Add') self.btAdd.Bind(wx.EVT_BUTTON, self._onAdd_constraint, id=self.btAdd.GetId()) self.btAdd.SetToolTipString("Add another constraint?") self.btAdd.Hide() text_hint = wx.StaticText(self, wx.ID_ANY, "Example: [M0][paramter] = M1.parameter") sizer_button.Add(text_hint, 0, wx.LEFT | wx.EXPAND | wx.ADJUST_MINSIZE, 10) sizer_button.Add(self.btAdd, 0, wx.LEFT | wx.EXPAND | wx.ADJUST_MINSIZE, 10) boxsizer1.Add(sizer_title, flag=wx.TOP | wx.BOTTOM, border=10) boxsizer1.Add(self.sizer_all_constraints, flag=wx.TOP | wx.BOTTOM, border=10) boxsizer1.Add(self.sizer_constraints, flag=wx.TOP | wx.BOTTOM, border=10) boxsizer1.Add(sizer_button, flag=wx.TOP | wx.BOTTOM, border=10) self.sizer2.Add(boxsizer1, 0, wx.EXPAND | wx.ALL, 10) def _fill_sizer_fit(self): """ Draw fit button """ self.sizer3.Clear(True) box_description = wx.StaticBox(self, wx.ID_ANY, "Fit ") boxsizer1 = wx.StaticBoxSizer(box_description, wx.VERTICAL) sizer_button = wx.BoxSizer(wx.HORIZONTAL) #Fit button self.btFit = wx.Button(self, self.ID_FIT, 'Fit', size=wx.DefaultSize) self.btFit.Bind(wx.EVT_BUTTON, self.onFit, id=self.btFit.GetId()) self.btFit.SetToolTipString("Perform fit.") #General Help button self.btHelp = wx.Button(self, wx.ID_HELP, 'HELP') self.btHelp.SetToolTipString("Simultaneous/Constrained Fitting help.") self.btHelp.Bind(wx.EVT_BUTTON, self._onHelp) #hint text on button line if self.batch_on: text = " Fit in Parallel all Data sets\n" text += "and model selected." else: text = " At least one set of model and data\n" text += " must be selected for fitting." text_hint = wx.StaticText(self, wx.ID_ANY, text) sizer_button.Add(text_hint) sizer_button.Add(self.btFit, 0, wx.LEFT | wx.ADJUST_MINSIZE, 10) sizer_button.Add(self.btHelp, 0, wx.LEFT | wx.ADJUST_MINSIZE, 10) boxsizer1.Add(sizer_button, flag=wx.TOP | wx.BOTTOM, border=10) self.sizer3.Add(boxsizer1, 0, wx.EXPAND | wx.ALL, 10)
[docs] def onRemove(self, event): """ Remove constraint fields """ if len(self.constraints_list) == 1: self.hide_constraint.SetValue(True) self._hide_constraint() return if len(self.constraints_list) == 0: return wx.CallAfter(self._remove_after, event.GetId()) #self._onAdd_constraint(None)
def _remove_after(self, id): for item in self.constraints_list: if id == item.btRemove.GetId(): self.sizer_constraints.Hide(item.sizer) item.sizer.Clear(True) self.sizer_constraints.Remove(item.sizer) self.constraints_list.remove(item) self.nb_constraint -= 1 self.sizer2.Layout() self.FitInside() break
[docs] def onFit(self, event): """ signal for fitting """ flag = False # check if the current page a simultaneous fit page or a batch page if self == self._manager.sim_page: flag = (self._manager.sim_page.uid == self.uid) ## making sure all parameters content a constraint if not self.batch_on and self.show_constraint.GetValue(): if not self._set_constraint(): return ## model was actually selected from this page to be fit if len(self.model_toFit) >= 1: self.manager._reset_schedule_problem(value=0) for item in self.model_list: if item[0].GetValue(): self.manager.schedule_for_fit(value=1, uid=item[2]) try: if not self.manager.onFit(uid=self.uid): return except: msg = "Select at least one parameter to fit in the FitPages." wx.PostEvent(self.parent.parent, StatusEvent(status=msg)) else: msg = "Select at least one model check box to fit " wx.PostEvent(self.parent.parent, StatusEvent(status=msg))
def _onHelp(self, event): """ Bring up the simultaneous Fitting Documentation whenever the HELP button is clicked. Calls DocumentationWindow with the path of the location within the documentation tree (after /doc/ ....". Note that when using old versions of Wx (before 2.9) and thus not the release version of installers, the help comes up at the top level of the file as webbrowser does not pass anything past the # to the browser when it is running "file:///...." :param evt: Triggers on clicking the help button """ _TreeLocation = "user/perspectives/fitting/fitting_help.html" _PageAnchor = "#simultaneous-fit-mode" _doc_viewer = DocumentationWindow(self, self.ID_DOC, _TreeLocation, _PageAnchor, "Simultaneous/Constrained Fitting Help")
[docs] def set_manager(self, manager): """ set panel manager :param manager: instance of plugin fitting """ self.manager = manager
[docs] def check_all_model_name(self, event=None): """ check all models names """ self.model_toFit = [] if self.cb1.GetValue() == True: for item in self.model_list: if item[0].IsEnabled(): item[0].SetValue(True) self.model_toFit.append(item) ## constraint info self._store_model() if not self.batch_on: ## display constraint fields if (self.show_constraint.GetValue() and len(self.constraints_list) == 0): self._show_all_constraint() self._show_constraint() else: for item in self.model_list: item[0].SetValue(False) if not self.batch_on: ##constraint info self._hide_constraint() self._update_easy_setup_cb() self.FitInside()
[docs] def check_model_name(self, event): """ Save information related to checkbox and their states """ self.model_toFit = [] cbox = event.GetEventObject() for item in self.model_list: if item[0].GetValue() == True: self.model_toFit.append(item) else: if item in self.model_toFit: self.model_toFit.remove(item) self.cb1.SetValue(False) ## display constraint fields if len(self.model_toFit) >= 1: self._store_model() if not self.batch_on and self.show_constraint.GetValue() and\ len(self.constraints_list) == 0: self._show_all_constraint() self._show_constraint() elif len(self.model_toFit) < 1: ##constraint info self._hide_constraint() self._update_easy_setup_cb() ## set the value of the main check button if len(self.model_list) == len(self.model_toFit): self.cb1.SetValue(True) self.FitInside() return else: self.cb1.SetValue(False) self.FitInside()
def _update_easy_setup_cb(self): """ Update easy setup combobox on selecting a model """ if self.model_cbox_left == None or self.model_cbox_right == None: return models = [(item[3].name, item[3]) for item in self.model_toFit] setComboBoxItems(self.model_cbox_left, models) setComboBoxItems(self.model_cbox_right, models) for item in self.constraints_list: setComboBoxItems(item[0], models) if self.model_cbox_left.GetSelection() == wx.NOT_FOUND: self.model_cbox_left.SetSelection(0) self.sizer2.Layout() def _store_model(self): """ Store selected model """ if len(self.model_toFit) < 1: return for item in self.model_toFit: model = item[3] page_id = item[2] self.constraint_dict[page_id] = model def _display_constraint(self, event): """ Show fields to add constraint """ if len(self.model_toFit) < 1: msg = "Select at least 1 model to add constraint " wx.PostEvent(self.parent.parent, StatusEvent(status=msg)) ## hide button self._hide_constraint() return if self.show_constraint.GetValue(): self._show_all_constraint() self._show_constraint() self.FitInside() return else: self._hide_constraint() return def _show_all_constraint(self): """ Show constraint fields """ box_description = wx.StaticBox(self, wx.ID_ANY, "Easy Setup ") boxsizer = wx.StaticBoxSizer(box_description, wx.HORIZONTAL) sizer_constraint = wx.BoxSizer(wx.HORIZONTAL) self.model_cbox_left = wx.ComboBox(self, wx.ID_ANY, style=wx.CB_READONLY) self.model_cbox_left.Clear() self.model_cbox_right = wx.ComboBox(self, wx.ID_ANY, style=wx.CB_READONLY) self.model_cbox_right.Clear() wx.EVT_COMBOBOX(self.model_cbox_left, wx.ID_ANY, self._on_select_modelcb) wx.EVT_COMBOBOX(self.model_cbox_right, wx.ID_ANY, self._on_select_modelcb) egal_txt = wx.StaticText(self, wx.ID_ANY, " = ") self.set_button = wx.Button(self, self.ID_SET_ALL, 'Set All') self.set_button.Bind(wx.EVT_BUTTON, self._on_set_all_equal, id=self.set_button.GetId()) set_tip = "Add constraints for all the adjustable parameters " set_tip += "(checked in FitPages) if exist." self.set_button.SetToolTipString(set_tip) self.set_button.Disable() for id, model in self.constraint_dict.iteritems(): ## check if all parameters have been selected for constraint ## then do not allow add constraint on parameters self.model_cbox_left.Append(str(model.name), model) self.model_cbox_left.Select(0) for id, model in self.constraint_dict.iteritems(): ## check if all parameters have been selected for constraint ## then do not allow add constraint on parameters self.model_cbox_right.Append(str(model.name), model) boxsizer.Add(self.model_cbox_left, flag=wx.RIGHT | wx.EXPAND, border=10) #boxsizer.Add(wx.StaticText(self, wx.ID_ANY, ".parameters"), # flag=wx.RIGHT | wx.EXPAND, border=5) boxsizer.Add(egal_txt, flag=wx.RIGHT | wx.EXPAND, border=5) boxsizer.Add(self.model_cbox_right, flag=wx.RIGHT | wx.EXPAND, border=10) #boxsizer.Add(wx.StaticText(self, wx.ID_ANY, ".parameters"), # flag=wx.RIGHT | wx.EXPAND, border=5) boxsizer.Add((20, -1)) boxsizer.Add(self.set_button, flag=wx.RIGHT | wx.EXPAND, border=5) sizer_constraint.Add(boxsizer, flag=wx.RIGHT | wx.EXPAND, border=5) self.sizer_all_constraints.Insert(before=0, item=sizer_constraint, flag=wx.TOP | wx.BOTTOM | wx.EXPAND, border=5) self.FitInside() def _on_select_modelcb(self, event): """ On select model left or right combobox """ event.Skip() flag = True if self.model_cbox_left.GetValue().strip() == '': flag = False if self.model_cbox_right.GetValue().strip() == '': flag = False if (self.model_cbox_left.GetValue() == self.model_cbox_right.GetValue()): flag = False self.set_button.Enable(flag) def _on_set_all_equal(self, event): """ On set button """ event.Skip() length = len(self.constraints_list) if length < 1: return param_list = [] param_listB = [] selection = self.model_cbox_left.GetCurrentSelection() model_left = self.model_cbox_left.GetValue() model = self.model_cbox_left.GetClientData(selection) selectionB = self.model_cbox_right.GetCurrentSelection() model_right = self.model_cbox_right.GetValue() modelB = self.model_cbox_right.GetClientData(selectionB) for id, dic_model in self.constraint_dict.iteritems(): if model == dic_model: param_list = self.page_finder[id].get_param2fit() if modelB == dic_model: param_listB = self.page_finder[id].get_param2fit() if len(param_list) > 0 and len(param_listB) > 0: break num_cbox = 0 has_param = False for param in param_list: num_cbox += 1 if param in param_listB: item = self.constraints_list[-1] item.model_cbox.SetStringSelection(model_left) self._on_select_model(None) item.param_cbox.Clear() item.param_cbox.Append(str(param), model) item.param_cbox.SetStringSelection(str(param)) item.constraint.SetValue(str(model_right + "." + str(param))) has_param = True if num_cbox == (len(param_list) + 1): break self._show_constraint() self.FitInside() if not has_param: msg = " There is no adjustable parameter (checked to fit)" msg += " either one of the models." wx.PostEvent(self.parent.parent, StatusEvent(info="warning", status=msg)) else: msg = " The constraints are added." wx.PostEvent(self.parent.parent, StatusEvent(info="info", status=msg)) def _show_constraint(self): """ Show constraint fields """ self.btAdd.Show(True) if len(self.constraints_list) != 0: nb_fit_param = 0 for id, model in self.constraint_dict.iteritems(): nb_fit_param += len(self.page_finder[id].get_param2fit()) ##Don't add anymore if len(self.constraints_list) == nb_fit_param: msg = "Cannot add another constraint. Maximum of number " msg += "Parameters name reached %s" % str(nb_fit_param) wx.PostEvent(self.parent.parent, StatusEvent(status=msg)) self.sizer_constraints.Layout() self.sizer2.Layout() return if len(self.model_toFit) < 1: msg = "Select at least 1 model to add constraint " wx.PostEvent(self.parent.parent, StatusEvent(status=msg)) self.sizer_constraints.Layout() self.sizer2.Layout() return sizer_constraint = wx.BoxSizer(wx.HORIZONTAL) # Model list model_cbox = wx.ComboBox(self, wx.ID_ANY, style=wx.CB_READONLY) model_cbox.Clear() for id, model in self.constraint_dict.iteritems(): ## check if all parameters have been selected for constraint ## then do not allow add constraint on parameters model_cbox.Append(str(model.name), model) wx.EVT_COMBOBOX(model_cbox, wx.ID_ANY, self._on_select_model) # Parameters in model param_cbox = wx.ComboBox(self, wx.ID_ANY, style=wx.CB_READONLY, size=(100, -1)) param_cbox.Hide() wx.EVT_COMBOBOX(param_cbox, wx.ID_ANY, self._on_select_param) egal_txt = wx.StaticText(self, wx.ID_ANY, " = ") # Parameter constraint constraint = wx.TextCtrl(self, wx.ID_ANY) # Remove button #btRemove = wx.Button(self, self.ID_REMOVE, 'Remove') btRemove = wx.Button(self, self._ids.next(), 'Remove') btRemove.Bind(wx.EVT_BUTTON, self.onRemove, id=btRemove.GetId()) btRemove.SetToolTipString("Remove constraint.") btRemove.Hide() # Hid the add button, if it exists if hasattr(self, "btAdd"): self.btAdd.Hide() sizer_constraint.Add((5, -1)) sizer_constraint.Add(model_cbox, flag=wx.RIGHT | wx.EXPAND, border=10) sizer_constraint.Add(param_cbox, flag=wx.RIGHT | wx.EXPAND, border=5) sizer_constraint.Add(egal_txt, flag=wx.RIGHT | wx.EXPAND, border=5) sizer_constraint.Add(constraint, flag=wx.RIGHT | wx.EXPAND, border=10) sizer_constraint.Add(btRemove, flag=wx.RIGHT | wx.EXPAND, border=10) self.sizer_constraints.Insert(before=self.nb_constraint, item=sizer_constraint, flag=wx.TOP | wx.BOTTOM | wx.EXPAND, border=5) c = ConstraintLine(model_cbox, param_cbox, egal_txt, constraint, btRemove, sizer_constraint) self.constraints_list.append(c) self.nb_constraint += 1 self.sizer_constraints.Layout() self.sizer2.Layout() self.Layout def _hide_constraint(self): """ hide buttons related constraint """ for id in self.page_finder.iterkeys(): self.page_finder[id].clear_model_param() self.nb_constraint = 0 self.constraint_dict = {} if hasattr(self, "btAdd"): self.btAdd.Hide() self._store_model() if self.model_cbox_left is not None: self.model_cbox_left.Clear() self.model_cbox_left = None if self.model_cbox_right is not None: self.model_cbox_right.Clear() self.model_cbox_right = None self.constraints_list = [] self.sizer_all_constraints.Clear(True) self.sizer_all_constraints.Layout() self.sizer_constraints.Clear(True) self.sizer_constraints.Layout() self.sizer2.Layout() self.Layout self.FitInside() def _on_select_model(self, event): """ fill combox box with list of parameters """ if not self.constraints_list: return ##This way PC/MAC both work, instead of using event.GetClientData(). model_cbox = self.constraints_list[-1].model_cbox n = model_cbox.GetCurrentSelection() if n == wx.NOT_FOUND: return model = model_cbox.GetClientData(n) param_list = [] for id, dic_model in self.constraint_dict.iteritems(): if model == dic_model: param_list = self.page_finder[id].get_param2fit() break param_cbox = self.constraints_list[-1].param_cbox param_cbox.Clear() ## insert only fittable paramaters for param in param_list: param_cbox.Append(str(param), model) param_cbox.Show(True) btRemove = self.constraints_list[-1].btRemove btRemove.Show(True) self.btAdd.Show(True) # self.Layout() self.FitInside() def _on_select_param(self, event): """ Store the appropriate constraint in the page_finder """ ##This way PC/MAC both work, instead of using event.GetClientData(). #n = self.param_cbox.GetCurrentSelection() #model = self.param_cbox.GetClientData(n) #param = event.GetString() if self.constraints_list: self.constraints_list[-1].egal_txt.Show(True) self.constraints_list[-1].constraint.Show(True) def _onAdd_constraint(self, event): """ Add another line for constraint """ if not self.show_constraint.GetValue(): msg = " Select Yes to add Constraint " wx.PostEvent(self.parent.parent, StatusEvent(status=msg)) return ## check that a constraint is added # before allow to add another constraint for item in self.constraints_list: if item.model_cbox.GetString(0) == "": msg = " Select a model Name! " wx.PostEvent(self.parent.parent, StatusEvent(status=msg)) return if item.param_cbox.GetString(0) == "": msg = " Select a parameter Name! " wx.PostEvent(self.parent.parent, StatusEvent(status=msg)) return if item.constraint.GetValue().lstrip().rstrip() == "": model = item.param_cbox.GetClientData( item.param_cbox.GetCurrentSelection()) if model != None: msg = " Enter a constraint for %s.%s! " % (model.name, item.param_cbox.GetString(0)) else: msg = " Enter a constraint" wx.PostEvent(self.parent.parent, StatusEvent(status=msg)) return ## some model or parameters can be constrained self._show_constraint() self.FitInside() def _set_constraint(self): """ get values from the constraint textcrtl ,parses them into model name parameter name and parameters values. store them in a list self.params .when when params is not empty set_model uses it to reset the appropriate model and its appropriates parameters """ for item in self.constraints_list: select0 = item.model_cbox.GetSelection() if select0 == wx.NOT_FOUND: continue model = item.model_cbox.GetClientData(select0) select1 = item.param_cbox.GetSelection() if select1 == wx.NOT_FOUND: continue param = item.param_cbox.GetString(select1) constraint = item.constraint.GetValue().lstrip().rstrip() if param.lstrip().rstrip() == "": param = None msg = " Constraint will be ignored!. missing parameters" msg += " in combobox to set constraint! " wx.PostEvent(self.parent.parent, StatusEvent(status=msg)) for id, value in self.constraint_dict.iteritems(): if model == value: if constraint == "": msg = " Constraint will be ignored!. missing value" msg += " in textcrtl to set constraint! " wx.PostEvent(self.parent.parent, StatusEvent(status=msg)) constraint = None if str(param) in self.page_finder[id].get_param2fit(): msg = " Checking constraint for parameter: %s ", param wx.PostEvent(self.parent.parent, StatusEvent(info="info", status=msg)) else: model_name = item[0].GetLabel() fitpage = self.page_finder[id].get_fit_tab_caption() msg = "All constrainted parameters must be set " msg += " adjustable: '%s.%s' " % (model_name, param) msg += "is NOT checked in '%s'. " % fitpage msg += " Please check it to fit or" msg += " remove the line of the constraint." wx.PostEvent(self.parent.parent, StatusEvent(info="error", status=msg)) return False for fid in self.page_finder[id].iterkeys(): # wrap in param/constraint in str() to remove unicode self.page_finder[id].set_model_param(str(param), str(constraint), fid=fid) break return True
[docs] def on_set_focus(self, event=None): """ The derivative class is on focus if implemented """ if self.parent is not None: if self.parent.parent is not None: wx.PostEvent(self.parent.parent, PanelOnFocusEvent(panel=self)) self.page_finder = self.parent._manager.get_page_finder()
[docs]def setComboBoxItems(cbox, items): assert isinstance(cbox, wx.ComboBox) selected = cbox.GetStringSelection() cbox.Clear() for k, (name, value) in enumerate(items): cbox.Append(name, value) cbox.SetStringSelection(selected)