Changeset 2eeda93 in sasview for src/sas


Ignore:
Timestamp:
Oct 19, 2018 7:25:25 AM (6 years ago)
Author:
Piotr Rozyczko <piotr.rozyczko@…>
Branches:
ESS_GUI, ESS_GUI_batch_fitting, ESS_GUI_bumps_abstraction, ESS_GUI_iss1116, ESS_GUI_opencl, ESS_GUI_ordering, ESS_GUI_sync_sascalc
Children:
b8dccb8
Parents:
10d57f6
Message:

Working version of Save/Load? Analysis. SASVIEW-983.
Changed the default behaviour of Category/Model? combos:
Selecting a category does not pre-select the first model now.

Location:
src/sas/qtgui
Files:
7 edited

Legend:

Unmodified
Added
Removed
  • src/sas/qtgui/MainWindow/DataExplorer.py

    r345b3b3 r2eeda93  
    235235            self.readProject(filename) 
    236236 
     237    def loadAnalysis(self): 
     238        """ 
     239        Called when the "Open Analysis" menu item chosen. 
     240        """ 
     241        kwargs = { 
     242            'parent'    : self, 
     243            'caption'   : 'Open Analysis', 
     244            'filter'    : 'Project (*.fitv);;All files (*.*)', 
     245            'options'   : QtWidgets.QFileDialog.DontUseNativeDialog 
     246        } 
     247        filename = QtWidgets.QFileDialog.getOpenFileName(**kwargs)[0] 
     248        if filename: 
     249            self.readAnalysis(filename) 
     250 
    237251    def saveProject(self): 
    238252        """ 
     
    256270        with open(filename, 'w') as outfile: 
    257271            self.saveDataToFile(outfile) 
     272 
     273    def saveAsAnalysisFile(self, tab_id=1): 
     274        """ 
     275        Show the save as... dialog and return the chosen filepath 
     276        """ 
     277        default_name = "FitPage"+str(tab_id)+".fitv" 
     278 
     279        wildcard = "fitv files (*.fitv)" 
     280        kwargs = { 
     281            'caption'   : 'Save As', 
     282            'directory' : default_name, 
     283            'filter'    : wildcard, 
     284            'parent'    : None, 
     285        } 
     286        # Query user for filename. 
     287        filename_tuple = QtWidgets.QFileDialog.getSaveFileName(**kwargs) 
     288        filename = filename_tuple[0] 
     289        return filename 
     290 
     291    def saveAnalysis(self, data, tab_id=1): 
     292        """ 
     293        Called when the "Save Analysis" menu item chosen. 
     294        """ 
     295        filename = self.saveAsAnalysisFile(tab_id) 
     296        if not filename: 
     297            return 
     298        _, extension = os.path.splitext(filename) 
     299        if not extension: 
     300            filename = '.'.join((filename, 'fitv')) 
     301        self.communicator.statusBarUpdateSignal.emit("Saving analysis... %s\n" % os.path.basename(filename)) 
     302 
     303        with open(filename, 'w') as outfile: 
     304            GuiUtils.saveData(outfile, data) 
     305 
     306        self.communicator.statusBarUpdateSignal.emit('Analysis saved.') 
    258307 
    259308    def allDataForModel(self, model): 
     
    275324        return all_data 
    276325 
    277     def saveDataToFile(self, outfile): 
    278         """ 
    279         Save every dataset to a json file 
     326    def getDataForID(self, id): 
     327        # return the dataset with the given ID 
     328        all_data = [] 
     329        for model in (self.model, self.theory_model): 
     330            for i in range(model.rowCount()): 
     331                properties = {} 
     332                item = model.item(i) 
     333                data = GuiUtils.dataFromItem(item) 
     334                if data is None: continue 
     335                if data.id != id: continue 
     336                # We found the dataset - save it. 
     337                filename = data.filename 
     338                is_checked = item.checkState() 
     339                properties['checked'] = is_checked 
     340                other_datas = GuiUtils.plotsFromFilename(filename, model) 
     341                # skip the main plot 
     342                other_datas = list(other_datas.values())[1:] 
     343                all_data = [data, properties, other_datas] 
     344                break 
     345        return all_data 
     346 
     347    def getAllData(self): 
     348        """ 
     349        converts all datasets into serializable dictionary 
    280350        """ 
    281351        data = self.allDataForModel(self.model) 
     
    289359                raise ValueError("Inconsistent data in Project file.") 
    290360            all_data[key] = value 
    291  
    292         # perspectives 
    293         current_perspective = self._perspective() 
    294         try: 
    295             perspective_dump = current_perspective.getCurrentStateAsXml() 
    296         except Exception: 
    297             ex = "State of " + current_perspective.windowTitle() + \ 
    298                 " cannot be saved." 
    299             logging.error(ex) 
    300         if perspective_dump is not None: 
    301             assert(isinstance(perspective_dump, dict)) 
    302             all_data['perspective'] = perspective_dump 
     361        return all_data 
     362 
     363    def saveDataToFile(self, outfile): 
     364        """ 
     365        Save every dataset to a json file 
     366        """ 
     367        all_data = self.getAllData() 
    303368        # save datas 
    304369        GuiUtils.saveData(outfile, all_data) 
    305         return 
    306370 
    307371    def readProject(self, filename): 
     
    311375        with open(filename, 'r') as infile: 
    312376            all_data = GuiUtils.readDataFromFile(infile) 
    313         # clear the model 
    314         self.model.clear() 
    315  
     377            items = self.updateModelFromData(all_data) 
     378 
     379        pass # debugger 
     380 
     381    def readAnalysis(self, filename): 
     382        """ 
     383        Read out a single dataset and fitpage from file 
     384        """ 
     385        with open(filename, 'r') as infile: 
     386            all_data = GuiUtils.readDataFromFile(infile) 
     387            # simulate full project structure 
     388            all_data_dict = {1:all_data['fit_data']} 
     389            items = self.updateModelFromData(all_data_dict) 
     390            # TODO: allow other perspectives 
     391            # send newly created item to its perspective 
     392            if len(items) > 0: 
     393                self.sendItemToPerspective(items[0]) 
     394            # Make the perspective read the rest of the read data 
     395            self._perspective().updateFromParameters(all_data['fit_params']) 
     396 
     397            pass 
     398 
     399    def updateModelFromData(self, data): 
     400        """ 
     401        Given data from analysis/project file, 
     402        create indices and populate data/theory models 
     403        """ 
     404        # model items for top level datasets 
     405        items = [] 
    316406        #self.model.beginResetModel() 
    317         for key, value in all_data.items(): 
     407        for key, value in data.items(): 
    318408            # key - cardinal number of dataset 
    319409            # value - main dataset, [dependant filesets] 
    320410            # add the main index 
     411            if not value: continue 
    321412            new_data = value[0] 
    322413            assert isinstance(new_data, (Data1D, Data2D)) 
     
    325416            new_item = GuiUtils.createModelItemWithPlot(new_data, new_data.filename) 
    326417            new_item.setCheckState(is_checked) 
     418            items.append(new_item) 
    327419            model = self.theory_model 
    328420            if new_data.is_data: 
     
    337429                assert isinstance(plot, (Data1D, Data2D)) 
    338430                GuiUtils.updateModelItemWithPlot(new_item, plot, plot.name) 
     431        return items 
    339432 
    340433    def deleteFile(self, event): 
     
    455548            retval = msgbox.exec_() 
    456549 
     550    def sendItemToPerspective(self, item): 
     551        """ 
     552        Send the passed item data to the current perspective and set the relevant notifiers 
     553        """ 
     554        # Set the signal handlers 
     555        self.communicator.updateModelFromPerspectiveSignal.connect(self.updateModelFromPerspective) 
     556        selected_items = [item] 
     557        # Notify the GuiManager about the send request 
     558        try: 
     559            self._perspective().setData(data_item=selected_items, is_batch=False) 
     560        except Exception as ex: 
     561            msg = "%s perspective returned the following message: \n%s\n" %(self._perspective().name, str(ex)) 
     562            logging.error(msg) 
     563            msg = str(ex) 
     564            msgbox = QtWidgets.QMessageBox() 
     565            msgbox.setIcon(QtWidgets.QMessageBox.Critical) 
     566            msgbox.setText(msg) 
     567            msgbox.setStandardButtons(QtWidgets.QMessageBox.Ok) 
     568            retval = msgbox.exec_() 
    457569 
    458570    def freezeCheckedData(self): 
  • src/sas/qtgui/MainWindow/GuiManager.py

    r6040cd7 r2eeda93  
    244244        if self._current_perspective: 
    245245            self._current_perspective.setClosable() 
    246             #self._workspace.workspace.removeSubWindow(self._current_perspective) 
    247246            self._current_perspective.close() 
    248247            self._workspace.workspace.removeSubWindow(self._current_perspective) 
     
    548547        """ 
    549548        """ 
    550         print("actionOpen_Analysis TRIGGERED") 
     549        self.filesWidget.loadAnalysis() 
    551550        pass 
    552551 
     
    561560        Menu File/Save Analysis 
    562561        """ 
    563         self.communicate.saveAnalysisSignal.emit() 
     562        per = self.perspective() 
     563        if not isinstance(per, FittingWindow): 
     564            return 
     565        # get fit page serialization 
     566        params = per.getSerializedFitpage() 
     567        data_id = per.currentTabDataId() 
     568        tab_id = per.currentTab.tab_id 
     569        data = self.filesWidget.getDataForID(data_id) 
     570        analysis = {} 
     571        analysis['fit_data'] = data 
     572        analysis['fit_params'] = params 
     573 
     574        self.filesWidget.saveAnalysis(analysis, tab_id) 
     575 
     576        pass 
    564577 
    565578    def actionQuit(self): 
     
    10231036        """ 
    10241037        self._workspace.actionReport.setEnabled(False) 
     1038        self._workspace.actionOpen_Analysis.setEnabled(False) 
     1039        self._workspace.actionSave_Analysis.setEnabled(False) 
     1040        if hasattr(perspective, 'isSerializable') and perspective.isSerializable(): 
     1041            self._workspace.actionOpen_Analysis.setEnabled(True) 
     1042            self._workspace.actionSave_Analysis.setEnabled(True) 
     1043 
    10251044        if isinstance(perspective, Perspectives.PERSPECTIVES["Fitting"]): 
    10261045            self.checkAnalysisOption(self._workspace.actionFitting) 
  • src/sas/qtgui/Perspectives/Fitting/FittingPerspective.py

    rb95d748 r2eeda93  
    112112    def onLatexCopy(self): 
    113113        self.currentTab.onCopyToClipboard("Latex") 
     114 
     115    def getSerializedFitpage(self): 
     116        # serialize current(active) fitpage 
     117        fitpage_state = self.currentTab.getFitPage() 
     118        fitpage_state += self.currentTab.getFitModel() 
     119        # put the text into dictionary 
     120        line_dict = {} 
     121        for line in fitpage_state: 
     122            #content = line.split(',') 
     123            if len(line) > 1: 
     124                line_dict[line[0]] = line[1:] 
     125        return line_dict 
     126 
     127    def currentTabDataId(self): 
     128        """ 
     129        Returns the data ID of the current tab 
     130        """ 
     131        tab_id = None 
     132        if self.currentTab.data: 
     133            tab_id = self.currentTab.data.id 
     134        return tab_id 
     135 
     136    def updateFromParameters(self, parameters): 
     137        """ 
     138        Pass the update parameters to the current fit page 
     139        """ 
     140        self.currentTab.createPageForParameters(parameters) 
    114141 
    115142    def closeEvent(self, event): 
     
    258285        return True 
    259286 
     287    def isSerializable(self): 
     288        """ 
     289        Tell the caller that this perspective writes its state 
     290        """ 
     291        return True 
     292 
    260293    def setData(self, data_item=None, is_batch=False): 
    261294        """ 
  • src/sas/qtgui/Perspectives/Fitting/FittingWidget.py

    r9a42ea1 r2eeda93  
    5252TAB_POLY = 3 
    5353CATEGORY_DEFAULT = "Choose category..." 
     54MODEL_DEFAULT = "Choose model..." 
    5455CATEGORY_STRUCTURE = "Structure Factor" 
    5556CATEGORY_CUSTOM = "Plugin Models" 
     
    566567        # Signals from other widgets 
    567568        self.communicate.customModelDirectoryChanged.connect(self.onCustomModelChange) 
    568         self.communicate.saveAnalysisSignal.connect(self.savePageState) 
    569         #self.communicate.loadAnalysisSignal.connect(self.loadPageState) 
    570569        self.smearing_widget.smearingChangedSignal.connect(self.onSmearingOptionsUpdate) 
    571570 
     
    10191018        model = self.cbModel.currentText() 
    10201019 
     1020        if model == MODEL_DEFAULT: 
     1021            # if the previous category was not the default, keep it. 
     1022            # Otherwise, just return 
     1023            if self._previous_model_index != 0: 
     1024                # We need to block signals, or else state changes on perceived unchanged conditions 
     1025                self.cbModel.blockSignals(True) 
     1026                self.cbModel.setCurrentIndex(self._previous_model_index) 
     1027                self.cbModel.blockSignals(False) 
     1028            return 
     1029 
    10211030        # Assure the control is active 
    10221031        if not self.cbModel.isEnabled(): 
     
    10261035            return 
    10271036        self.chkMagnetism.setEnabled(self.canHaveMagnetism()) 
     1037        self.chkMagnetism.setEnabled(self.canHaveMagnetism()) 
    10281038        self.tabFitting.setTabEnabled(TAB_MAGNETISM, self.chkMagnetism.isChecked() and self.canHaveMagnetism()) 
     1039        self._previous_model_index = self.cbModel.currentIndex() 
    10291040 
    10301041        # Reset parameters to fit 
     
    12001211            self._model_model.clear() 
    12011212            return 
    1202  
     1213        # Wipe out the parameter model 
     1214        self._model_model.clear() 
    12031215        # Safely clear and enable the model combo 
    12041216        self.cbModel.blockSignals(True) 
     
    12121224        model_list = self.master_category_dict[category] 
    12131225        # Populate the models combobox 
     1226        self.cbModel.blockSignals(True) 
     1227        self.cbModel.addItem(MODEL_DEFAULT) 
    12141228        self.cbModel.addItems(sorted([model for (model, _) in model_list])) 
     1229        self.cbModel.blockSignals(False) 
    12151230 
    12161231    def onPolyModelChange(self, top, bottom): 
     
    33663381        return report_logic.reportList() 
    33673382 
    3368     def savePageState(self): 
    3369         """ 
    3370         Create and serialize local PageState 
    3371         """ 
    3372         filepath = self.saveAsAnalysisFile() 
    3373         if filepath is None or filepath == "": 
    3374             return 
    3375  
    3376         fitpage_state = self.getFitPage() 
    3377         fitpage_state += self.getFitModel() 
    3378  
    3379         with open(filepath, 'w') as statefile: 
    3380             for line in fitpage_state: 
    3381                 statefile.write(str(line)) 
    3382  
    3383         self.communicate.statusBarUpdateSignal.emit('Analysis saved.') 
    3384  
    3385     def saveAsAnalysisFile(self): 
    3386         """ 
    3387         Show the save as... dialog and return the chosen filepath 
    3388         """ 
    3389         default_name = "FitPage"+str(self.tab_id)+".fitv" 
    3390  
    3391         wildcard = "fitv files (*.fitv)" 
    3392         kwargs = { 
    3393             'caption'   : 'Save As', 
    3394             'directory' : default_name, 
    3395             'filter'    : wildcard, 
    3396             'parent'    : None, 
    3397         } 
    3398         # Query user for filename. 
    3399         filename_tuple = QtWidgets.QFileDialog.getSaveFileName(**kwargs) 
    3400         filename = filename_tuple[0] 
    3401         return filename 
    3402  
    34033383    def loadPageStateCallback(self,state=None, datainfo=None, format=None): 
    34043384        """ 
     
    35193499        """ 
    35203500        param_list = [] 
     3501        if self.kernel_module is None: 
     3502            return param_list 
     3503 
    35213504        param_list.append(['model_name', str(self.cbModel.currentText())]) 
    35223505 
     
    36233606        cb_text = cb.text() 
    36243607 
    3625         context = {} 
    36263608        lines = cb_text.split(':') 
    36273609        if lines[0] != 'sasview_parameter_values': 
     
    36353617                line_dict[content[0]] = content[1:] 
    36363618 
     3619        self.updatePageWithParameters(line_dict) 
     3620 
     3621    def createPageForParameters(self, line_dict): 
     3622        """ 
     3623        Sets up page with requested model/str factor 
     3624        and fills it up with sent parameters 
     3625        """ 
     3626        if 'fitpage_category' in line_dict: 
     3627            self.cbCategory.setCurrentIndex(self.cbCategory.findText(line_dict['fitpage_category'][0])) 
     3628        if 'fitpage_model' in line_dict: 
     3629            self.cbModel.setCurrentIndex(self.cbModel.findText(line_dict['fitpage_model'][0])) 
     3630        if 'fitpage_structure' in line_dict: 
     3631            self.cbStructureFactor.setCurrentIndex(self.cbStructureFactor.findText(line_dict['fitpage_structure'][0])) 
     3632 
     3633        # Now that the page is ready for parameters, fill it up 
     3634        self.updatePageWithParameters(line_dict) 
     3635 
     3636    def updatePageWithParameters(self, line_dict): 
     3637        """ 
     3638        Update FitPage with parameters in line_dict 
     3639        """ 
     3640        if 'model_name' not in line_dict.keys(): 
     3641            return 
    36373642        model = line_dict['model_name'][0] 
    3638  
    3639         if 'model_name' not in line_dict.keys(): 
    3640             return False 
     3643        context = {} 
    36413644 
    36423645        if 'multiplicity' in line_dict.keys(): 
  • src/sas/qtgui/Perspectives/Fitting/UnitTesting/ComplexConstraintTest.py

    r725d9c06 r2eeda93  
    3434        category_index = self.tab1.cbCategory.findText("Shape Independent") 
    3535        self.tab1.cbCategory.setCurrentIndex(category_index) 
     36        model_index = self.tab1.cbModel.findText("be_polyelectrolyte") 
     37        self.tab1.cbModel.setCurrentIndex(model_index) 
     38 
    3639        category_index = self.tab2.cbCategory.findText("Cylinder") 
    3740        self.tab2.cbCategory.setCurrentIndex(category_index) 
     41        model_index = self.tab2.cbModel.findText("barbell") 
     42        self.tab2.cbModel.setCurrentIndex(model_index) 
    3843 
    3944        tabs = [self.tab1, self.tab2] 
  • src/sas/qtgui/Perspectives/Fitting/UnitTesting/FittingWidgetTest.py

    rd2007a8 r2eeda93  
    175175        category_index = self.widget.cbCategory.findText("Shape Independent") 
    176176        self.widget.cbCategory.setCurrentIndex(category_index) 
     177        model_index = self.widget.cbModel.findText("be_polyelectrolyte") 
     178        self.widget.cbModel.setCurrentIndex(model_index) 
    177179 
    178180        # test the model combo content 
    179         self.assertEqual(self.widget.cbModel.count(), 29) 
     181        self.assertEqual(self.widget.cbModel.count(), 30) 
    180182 
    181183        # Try to change back to default 
     
    201203        category_index = self.widget.cbCategory.findText("Shape Independent") 
    202204        self.widget.cbCategory.setCurrentIndex(category_index) 
     205        model_index = self.widget.cbModel.findText("be_polyelectrolyte") 
     206        self.widget.cbModel.setCurrentIndex(model_index) 
    203207 
    204208        # check the enablement of controls 
     
    215219        #  
    216220        # Now change the model 
    217         self.widget.cbModel.setCurrentIndex(3) 
     221        self.widget.cbModel.setCurrentIndex(4) 
    218222        self.assertEqual(self.widget.cbModel.currentText(),'dab') 
    219223 
     
    226230        self.widget.data_is_loaded = True 
    227231        # Reset the sasmodel index 
    228         self.widget.cbModel.setCurrentIndex(1) 
     232        self.widget.cbModel.setCurrentIndex(2) 
    229233        self.assertEqual(self.widget.cbModel.currentText(),'broad_peak') 
    230234 
     
    377381        category_index = self.widget.cbCategory.findText("Shape Independent") 
    378382        self.widget.cbCategory.setCurrentIndex(category_index) 
     383        model_index = self.widget.cbModel.findText("be_polyelectrolyte") 
     384        self.widget.cbModel.setCurrentIndex(model_index) 
     385 
    379386        # Check the poly model 
    380387        self.assertEqual(self.widget._poly_model.rowCount(), 0) 
     
    383390        # Change the category index so we have a model available 
    384391        self.widget.cbCategory.setCurrentIndex(2) 
     392        self.widget.cbModel.setCurrentIndex(1) 
    385393 
    386394        # Check the poly model 
     
    556564        category_index = self.widget.cbCategory.findText("Sphere") 
    557565        self.widget.cbCategory.setCurrentIndex(category_index) 
     566        model_index = self.widget.cbModel.findText("adsorbed_layer") 
     567        self.widget.cbModel.setCurrentIndex(model_index) 
    558568 
    559569        # Check the magnetic model 
     
    634644        category_index = self.widget.cbCategory.findText("Sphere") 
    635645        self.widget.cbCategory.setCurrentIndex(category_index) 
     646        model_index = self.widget.cbModel.findText("adsorbed_layer") 
     647        self.widget.cbModel.setCurrentIndex(model_index) 
    636648 
    637649        # Check the enablement/text 
     
    973985        category_index = self.widget.cbCategory.findText("Sphere") 
    974986        self.widget.cbCategory.setCurrentIndex(category_index) 
     987        model_index = self.widget.cbModel.findText("adsorbed_layer") 
     988        self.widget.cbModel.setCurrentIndex(model_index) 
    975989        self.widget.main_params_to_fit = ['scale'] 
    976990 
     
    9861000        self.assertListEqual(fp.main_params_to_fit, ['scale']) 
    9871001 
    988     def testPushFitPage(self): 
     1002    def notestPushFitPage(self): 
    9891003        """ 
    9901004        Push current state of fitpage onto stack 
     
    9971011        self.widget.data = item 
    9981012        category_index = self.widget.cbCategory.findText("Sphere") 
     1013        model_index = self.widget.cbModel.findText("adsorbed_layer") 
     1014        self.widget.cbModel.setCurrentIndex(model_index) 
    9991015 
    10001016        # Asses the initial state of stack 
  • src/sas/qtgui/Utilities/GuiUtils.py

    r345b3b3 r2eeda93  
    270270    sendDataToGridSignal = QtCore.pyqtSignal(list) 
    271271 
    272     # Action Save Analysis triggered 
    273     saveAnalysisSignal = QtCore.pyqtSignal() 
    274  
    275272    # Mask Editor requested 
    276273    maskEditorSignal = QtCore.pyqtSignal(Data2D) 
     
    577574    if isinstance(data.process, list) and data.process: 
    578575        for process in data.process: 
     576            if process is None: 
     577                continue 
    579578            process_date = process.date 
    580579            process_date_item = QtGui.QStandardItem("Date: " + process_date) 
Note: See TracChangeset for help on using the changeset viewer.