X-Git-Url: https://codewiz.org/gitweb?a=blobdiff_plain;f=wizard%2FBToolchainPage.py;h=041a8069a8c86cb62034ae8d33d9daa4da984d1f;hb=HEAD;hp=4b2b4c5e21f1063833c0456d9402556eebd87e2a;hpb=5423f0cee8fe3413d7e2f756e66a111c4dd8eb2a;p=bertos.git diff --git a/wizard/BToolchainPage.py b/wizard/BToolchainPage.py index 4b2b4c5e..041a8069 100644 --- a/wizard/BToolchainPage.py +++ b/wizard/BToolchainPage.py @@ -1,110 +1,299 @@ #!/usr/bin/env python # encoding: utf-8 # +# This file is part of BeRTOS. +# +# Bertos is free software; you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation; either version 2 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +# +# As a special exception, you may use this file as part of a free software +# library without restriction. Specifically, if other files instantiate +# templates or use macros or inline functions from this file, or you compile +# this file and link it with other files to produce an executable, this +# file does not by itself cause the resulting executable to be covered by +# the GNU General Public License. This exception does not however +# invalidate any other reasons why the executable file might be covered by +# the GNU General Public License. +# # Copyright 2008 Develer S.r.l. (http://www.develer.com/) -# All rights reserved. # -# $Id:$ # # Author: Lorenzo Berni # +import os +import collections + from BWizardPage import * +from BCreationPage import BCreationPage + +import BToolchainSearch import bertos_utils +import qvariant_converter + +from toolchain_manager import ToolchainManager + +from const import * class BToolchainPage(BWizardPage): - + """ + Page of the wizard that permits to choose the toolchain to use for the + project. + """ + def __init__(self): - BWizardPage.__init__(self, "toolchain_select.ui") + BWizardPage.__init__(self, UI_LOCATION + "/toolchain_select.ui") self.setTitle(self.tr("Select toolchain")) - self._setupUi() - self._populateDirList() - self._connectSignals() - - def _setupUi(self): - path = self._settingsRetrieve("path_search").toBool() - if not path is None: - self.pageContent.pathBox.setChecked(path) - else: - self.pageContent.pathBox.setChecked(False) - customPath = self._settingsRetrieve("custom_dir").toBool() - if not customPath is None: - self.pageContent.customDirBox.setChecked(customPath) - else: - self.pageContent.customDirBox.setChecked(False) - self._updateUi() - - def _updateUi(self): - if self.pageContent.customDirBox.isChecked(): - self._enableCustomDir() + self.setSubTitle(self.tr("You can look for more toolchains in your system by pressing the \"Search\" button, or manually add them with the \"+\" button")) + self._valid_items = [] + self._toolchain_manager = ToolchainManager() + + ## Overloaded QWizardPage methods. ## + + def isComplete(self): + """ + Overload of the QWizard isComplete method. + """ + if self.pageContent.toolchainList.currentRow() != -1: + self.setProjectInfo("TOOLCHAIN", + qvariant_converter.getStringDict(self.pageContent.toolchainList.currentItem().data(Qt.UserRole))) + return True else: - self._disableCustomDir() - if self.pageContent.pathBox.isChecked() or (self.pageContent.customDirBox.isChecked() and self.pageContent.customDirList.count() != 0): - self.pageContent.doSearchButton.setEnabled(True) + return False + + def nextId(self): + """ + Overload of the QWizardPage nextId method. + """ + # Route to Output page if it's a predefined easy project. + if self.projectInfo("PROJECT_FROM_PRESET") and self.projectInfo("BASE_MODE"): + return self.wizard().pageIndex(BCreationPage) else: - self.pageContent.doSearchButton.setEnabled(False) - - def _populateDirList(self): - search_dir_list = self._settingsRetrieve("search_dir_list").toList() - for element in search_dir_list: - item = QListWidgetItem(element.toString()) - item.setData(Qt.UserRole, element) - self.pageContent.customDirList.addItem(item) - - def _disableCustomDir(self): - self.pageContent.customDirList.setEnabled(False) - self.pageContent.addDirButton.setEnabled(False) - self.pageContent.removeDirButton.setEnabled(False) - - def _enableCustomDir(self): - self.pageContent.customDirList.setEnabled(True) - self.pageContent.addDirButton.setEnabled(True) - self.pageContent.removeDirButton.setEnabled(True) - + return QWizardPage.nextId(self) + + #### + + ## Overloaded BWizardPage methods. ## + + def setupUi(self): + """ + Sets up the user interface. + """ + self.pageContent.infoLabel.setVisible(False) + + def connectSignals(self): + """ + Connects the signals with the related slots. + """ + self.connect(self.pageContent.toolchainList, SIGNAL("currentItemChanged(QListWidgetItem *, QListWidgetItem*)"), self.selectionChanged) + self.connect(self.pageContent.addButton, SIGNAL("clicked()"), self.addToolchain) + self.connect(self.pageContent.removeButton, SIGNAL("clicked()"), self.removeToolchain) + self.connect(self.pageContent.searchButton, SIGNAL("clicked()"), self.searchToolchain) + self.connect(self.pageContent.checkButton, SIGNAL("clicked()"), self.validateAllToolchains) + + def reloadData(self, previous_id=None): + """ + Overload of the BWizard reloadData method. + """ + if previous_id is None or previous_id < self.wizard().currentId(): + self._clearList() + self.setupUi() + self._populateToolchainList() + if len(self._valid_items) >= 1: + self.pageContent.toolchainList.setCurrentItem(self._valid_items[0]) + + #### + + ## Slots ## + + def selectionChanged(self): + """ + Slot called when the user click on an entry of the toolchain list. + """ + if self.pageContent.toolchainList.currentRow() != -1: + infos = collections.defaultdict(lambda: unicode("not defined")) + infos.update(qvariant_converter.getStringDict(self.pageContent.toolchainList.currentItem().data(Qt.UserRole))) + self.pageContent.infoLabel.setText("GCC " + infos["version"] + " (" + infos["build"] + ")\nTarget: " + infos["target"] + "\nPath: " + os.path.normpath(infos["path"])) + self.pageContent.infoLabel.setVisible(True) + if self.isDefaultToolchain(infos): + self.disableRemoveButton() + else: + self.enableRemoveButton() + self.emit(SIGNAL("completeChanged()")) + + def addToolchain(self): + """ + Slot called when the user adds manually a toolchain. + """ + sel_toolchain = unicode(QFileDialog.getOpenFileName(self, self.tr("Choose the toolchain"), "")) + if sel_toolchain != "": + item = QListWidgetItem(sel_toolchain) + item.setData(Qt.UserRole, qvariant_converter.convertStringDict({"path": sel_toolchain})) + self.pageContent.toolchainList.addItem(item) + self._toolchain_manager.addToolchain(sel_toolchain) + + def removeToolchain(self): + """ + Slot called when the user removes manually a toolchain. + """ + if self.pageContent.toolchainList.currentRow() != -1: + item = self.pageContent.toolchainList.takeItem(self.pageContent.toolchainList.currentRow()) + toolchain = qvariant_converter.getStringDict(item.data(Qt.UserRole))["path"] + self._toolchain_manager.removeToolchain(toolchain) + + def searchToolchain(self): + """ + Slot called when the user clicks on the 'search' button. It opens the + toolchain search dialog. + """ + search = BToolchainSearch.BToolchainSearch() + self.connect(search, SIGNAL("accepted()"), self._search) + search.exec_() + + def validateAllToolchains(self): + """ + Slot called when the user clicks on the validate button. It starts the + toolchain validation procedure for all the toolchains. + """ + try: + QApplication.instance().setOverrideCursor(Qt.WaitCursor) + for i in range(self.pageContent.toolchainList.count()): + data = qvariant_converter.getStringDict(self.pageContent.toolchainList.item(i).data(Qt.UserRole)) + self.validateToolchain(data["path"]) + self._populateToolchainList() + finally: + QApplication.instance().restoreOverrideCursor() + + #### + + def _populateToolchainList(self): + """ + Fills the toolchain list with the toolchains stored in the QSettings. + """ + self.pageContent.toolchainList.clear() + self._valid_items = [] + toolchains = self._toolchain_manager.predefined_toolchains + self._toolchain_manager.toolchains + toolchain_dict = { + 'valid': [], + 'non-valid': [], + 'unknown': [], + 'unverified': [], + } + for key, value in toolchains: + if os.path.exists(key): + item_data = {"path":key} + if value: + item_data.update(value) + if value is not None: + k, data = self.validateToolchain(key) + item_data.update(data) + toolchain_dict[k].append(item_data) + else: + toolchain_dict["unverified"].append(item_data) + self._processItems(toolchain_dict["valid"], self._validItem) + self._processItems(toolchain_dict["non-valid"], self._invalidItem) + self._processItems(toolchain_dict["unknown"], self._unknownItem) + self._processItems(toolchain_dict["unverified"]) + + def _processItems(self, item_list, procedure=lambda x: None): + sel_toolchain = self.projectInfo("TOOLCHAIN") + for item_data in item_list: + item = QListWidgetItem(item_data["path"]) + item.setData(Qt.UserRole, qvariant_converter.convertStringDict(item_data)) + self.pageContent.toolchainList.addItem(item) + procedure(item) + if sel_toolchain and sel_toolchain["path"] == item_data["path"]: + self.pageContent.toolchainList.setCurrentItem(item) + + def currentToolchain(self): + selected_toolchain = qvariant_converter.getStringDict(self.pageContent.toolchainList.currentItem().data(Qt.UserRole)) + return selected_toolchain + def _clearList(self): + """ + Removes all the toolchain from the list. + """ self.pageContent.toolchainList.clear() + + def _search(self): + """ + Searches for toolchains in the stored directories, and stores them in the + QSettings. + """ + dir_list = self.searchDirList() + if self.pathSearch(): + dir_list += [element for element in bertos_utils.getSystemPath()] + _toolchain_dict = self._toolchain_manager.storedToolchainDict() + toolchain_list = bertos_utils.findToolchains(dir_list) + for toolchain in toolchain_list: + self._toolchain_manager.addToolchain(toolchain, _toolchain_dict.get(toolchain, False)) + self._populateToolchainList() + self.showMessage(self.tr("Toolchain search result."), self.tr("%1 toolchains found").arg(len(toolchain_list))) + + def _validItem(self, item): + """ + Sets the item at index as a valid item and associates the given info to it. + """ + data = qvariant_converter.getStringDict(item.data(Qt.UserRole)) + item.setIcon(QIcon(":/images/ok.png")) + self._valid_items.append(item) + if "version" in data and "target" in data: + item.setText("GCC " + data["version"] + " - " + data["target"].strip()) + + def _invalidItem(self, item): + data = qvariant_converter.getStringDict(item.data(Qt.UserRole)) + item.setIcon(QIcon(":/images/warning.png")) + if "version" in data and "target" in data: + item.setText("GCC " + data["version"] + " - " + data["target"].strip()) + + def _unknownItem(self, item): + """ + Sets the item at index as an invalid item. + """ + item.setIcon(QIcon(":/images/error.png")) + + def validateToolchain(self, filename): + """ + Toolchain validation procedure. + """ + info = self._toolchain_manager.validateToolchain(filename) + if info: + needed = self.projectInfo("CPU_INFOS") + if "target" in info and info["target"].find(needed["TOOLCHAIN"]) != -1: + return 'valid', info + else: + return 'non-valid', info + else: + return 'unknown', {'path': filename} - def _connectSignals(self): - self.connect(self.pageContent.pathBox, SIGNAL("clicked()"), self._updateUi) - self.connect(self.pageContent.customDirBox, SIGNAL("clicked()"), self._updateUi) - self.connect(self.pageContent.searchButton, SIGNAL("clicked()"), self.toSearchSubpage) - self.connect(self.pageContent.cancelButton, SIGNAL("clicked()"), self.toSelectionSubpage) - self.connect(self.pageContent.doSearchButton, SIGNAL("clicked()"), self.doSearch) - self.connect(self.pageContent.addDirButton, SIGNAL("clicked()"), self.addDir) - self.connect(self.pageContent.removeDirButton, SIGNAL("clicked()"), self.removeDir) - - def toSearchSubpage(self): - self.pageContent.pageSelector.setCurrentIndex(1) - - def toSelectionSubpage(self): - self.pageContent.pageSelector.setCurrentIndex(0) - - def doSearch(self): - path = [] - if self.pageContent.pathBox.isChecked(): - path += bertos_utils.getSystemPath() - if self.pageContent.customDirBox.isChecked(): - for element in range(self.pageContent.customDirList.count()): - path += [unicode(self.pageContent.customDirList.item(element).data(Qt.UserRole).toString())] - toolchains = bertos_utils.findToolchains(path) - self._clearList() - for toolchain in toolchains: - self.pageContent.toolchainList.addItem(QListWidgetItem(toolchain)) - self.toSelectionSubpage() + def isDefaultToolchain(self, toolchain): + """ + Returns True if the given toolchain is one of the default toolchains. + """ + return toolchain["path"] in self._toolchain_manager._predefined_toolchain_set - def addDir(self): - directory = QFileDialog.getExistingDirectory(self, self.tr("Open Directory"), "", QFileDialog.ShowDirsOnly) - if not directory.isEmpty(): - item = QListWidgetItem(directory) - item.setData(Qt.UserRole, QVariant(directory)) - self.pageContent.customDirList.addItem(item) - search_dir_list = self._settingsRetrieve("search_dir_list").toList() - search_dir_list = set([d.toString() for d in search_dir_list] + [directory]) - self._settingsStore("search_dir_list", list(search_dir_list)) + def disableRemoveButton(self): + """ + Disable the remove button. + """ + self.pageContent.removeButton.setEnabled(False) - def removeDir(self): - item = self.pageContent.customDirList.takeItem(self.pageContent.customDirList.currentRow()) - search_dir_list = self._settingsRetrieve("search_dir_list").toList() - search_dir_list = set([d.toString() for d in search_dir_list]) - search_dir_list.remove(item.data(Qt.UserRole).toString()) - self._settingsStore("search_dir_list", list(search_dir_list)) \ No newline at end of file + def enableRemoveButton(self): + """ + Enable the remove button. + """ + self.pageContent.removeButton.setEnabled(True) + + def currentItem(self): + return self.pageContent.toolchainList.currentItem()