Source code for sas.qtgui.Calculators.DensityPanel

# global
import logging
import functools
from PyQt5 import QtCore
from PyQt5 import QtGui
from PyQt5 import QtWidgets

from periodictable import formula as Formula

from sas.qtgui.Utilities.GuiUtils import FormulaValidator
from sas.qtgui.UI import main_resources_rc
from sas.qtgui.Utilities.GuiUtils import HELP_DIRECTORY_LOCATION

# Local UI
from sas.qtgui.Calculators.UI.DensityPanel import Ui_DensityPanel

from sas.qtgui.Utilities.GuiUtils import enum
from sas.qtgui.Utilities.GuiUtils import formatNumber

MODEL = enum(
    'MOLECULAR_FORMULA',
    'MOLAR_MASS',
    'MOLAR_VOLUME',
    'MASS_DENSITY',
)

MODES = enum(
    'VOLUME_TO_DENSITY',
    'DENSITY_TO_VOLUME',
)

[docs]def toMolarMass(formula): AVOGADRO = 6.02214129e23 try: f = Formula(str(formula)) return "%g" % (f.molecular_mass * AVOGADRO) except: return ""
class DensityPanel(QtWidgets.QDialog): def __init__(self, parent=None): super(DensityPanel, self).__init__() self.mode = None self.manager = parent self.setupUi() # disable the context help icon self.setWindowFlags(self.windowFlags() & ~QtCore.Qt.WindowContextHelpButtonHint) self.setupModel() self.setupMapper() def setupUi(self): self.ui = Ui_DensityPanel() self.ui.setupUi(self) #self.setFixedSize(self.minimumSizeHint()) self.resize(self.minimumSizeHint()) # set validators #self.ui.editMolecularFormula.setValidator(FormulaValidator(self.ui.editMolecularFormula)) rx = QtCore.QRegExp("[+\-]?(?:0|[1-9]\d*)(?:\.\d*)?(?:[eE][+\-]?\d+)?") self.ui.editMolarVolume.setValidator(QtGui.QRegExpValidator(rx, self.ui.editMolarVolume)) self.ui.editMassDensity.setValidator(QtGui.QRegExpValidator(rx, self.ui.editMassDensity)) # signals self.ui.editMolarVolume.textEdited.connect(functools.partial(self.setMode, MODES.VOLUME_TO_DENSITY)) self.ui.editMassDensity.textEdited.connect(functools.partial(self.setMode, MODES.DENSITY_TO_VOLUME)) self.ui.buttonBox.button(QtWidgets.QDialogButtonBox.Reset).clicked.connect(self.modelReset) self.ui.buttonBox.button(QtWidgets.QDialogButtonBox.Help).clicked.connect(self.displayHelp) def setupModel(self): self.model = QtGui.QStandardItemModel(self) self.model.setItem(MODEL.MOLECULAR_FORMULA, QtGui.QStandardItem()) self.model.setItem(MODEL.MOLAR_MASS , QtGui.QStandardItem()) self.model.setItem(MODEL.MOLAR_VOLUME , QtGui.QStandardItem()) self.model.setItem(MODEL.MASS_DENSITY , QtGui.QStandardItem()) self.model.dataChanged.connect(self.dataChanged) self.ui.editMolarVolume.textEdited.connect(self.volumeChanged) self.ui.editMassDensity.textEdited.connect(self.massChanged) self.ui.editMolecularFormula.textEdited.connect(self.formulaChanged) self.modelReset() def setupMapper(self): self.mapper = QtWidgets.QDataWidgetMapper(self) self.mapper.setModel(self.model) self.mapper.setOrientation(QtCore.Qt.Vertical) self.mapper.addMapping(self.ui.editMolecularFormula, MODEL.MOLECULAR_FORMULA) self.mapper.addMapping(self.ui.editMolarMass , MODEL.MOLAR_MASS) self.mapper.addMapping(self.ui.editMolarVolume , MODEL.MOLAR_VOLUME) self.mapper.addMapping(self.ui.editMassDensity , MODEL.MASS_DENSITY) self.mapper.toFirst() def dataChanged(self, top, bottom): for index in range(top.row(), bottom.row() + 1): if index == MODEL.MOLECULAR_FORMULA: molarMass = toMolarMass(self.model.item(MODEL.MOLECULAR_FORMULA).text()) molarMass = formatNumber(molarMass, high=True) self.model.item(MODEL.MOLAR_MASS).setText(molarMass) if self.mode == MODES.VOLUME_TO_DENSITY: self._updateDensity() elif self.mode == MODES.DENSITY_TO_VOLUME: self._updateVolume() elif index == MODEL.MOLAR_VOLUME and self.mode == MODES.VOLUME_TO_DENSITY: self._updateDensity() elif index == MODEL.MASS_DENSITY and self.mode == MODES.DENSITY_TO_VOLUME: self._updateVolume() def volumeChanged(self, current_text): try: molarMass = float(toMolarMass(self.model.item(MODEL.MOLECULAR_FORMULA).text())) molarVolume = float(current_text) molarDensity = molarMass / molarVolume molarDensity = formatNumber(molarDensity, high=True) self.model.item(MODEL.MASS_DENSITY).setText(str(molarDensity)) except (ArithmeticError, ValueError): self.model.item(MODEL.MASS_DENSITY).setText("") def massChanged(self, current_text): try: molarMass = float(toMolarMass(self.model.item(MODEL.MOLECULAR_FORMULA).text())) molarDensity = float(current_text) molarVolume = molarMass / molarDensity molarVolume = formatNumber(molarVolume, high=True) self.model.item(MODEL.MOLAR_VOLUME).setText(str(molarVolume)) except (ArithmeticError, ValueError): self.model.item(MODEL.MOLAR_VOLUME).setText("") def formulaChanged(self, current_text): try: molarMass = toMolarMass(current_text) # if this doesn't fail, update the model item for formula # so related values can get recomputed self.model.item(MODEL.MOLECULAR_FORMULA).setText(current_text) except (ArithmeticError, ValueError): self.model.item(MODEL.MOLAR_VOLUME).setText("") def setMode(self, mode): self.mode = mode def _updateDensity(self): try: molarMass = float(toMolarMass(self.model.item(MODEL.MOLECULAR_FORMULA).text())) molarVolume = float(self.model.item(MODEL.MOLAR_VOLUME).text()) molarDensity = molarMass / molarVolume molarDensity = formatNumber(molarDensity, high=True) self.model.item(MODEL.MASS_DENSITY).setText(str(molarDensity)) except (ArithmeticError, ValueError): self.model.item(MODEL.MASS_DENSITY).setText("") def _updateVolume(self): try: molarMass = float(toMolarMass(self.model.item(MODEL.MOLECULAR_FORMULA).text())) molarDensity = float(self.model.item(MODEL.MASS_DENSITY).text()) molarVolume = molarMass / molarDensity molarVolume = formatNumber(molarVolume, high=True) self.model.item(MODEL.MOLAR_VOLUME).setText(str(molarVolume)) except (ArithmeticError, ValueError): self.model.item(MODEL.MOLAR_VOLUME).setText("") def modelReset(self): try: self.setMode(None) self.model.item(MODEL.MOLECULAR_FORMULA).setText("H2O") self.model.item(MODEL.MOLAR_VOLUME ).setText("") self.model.item(MODEL.MASS_DENSITY ).setText("") finally: pass def displayHelp(self): location = "/user/qtgui/Calculators/density_calculator_help.html" self.manager.showHelp(location)