Source code for sas.perspectives.invariant.invariant_details

"""
    Invariant panel
"""
import wx
import sys

from sas.guiframe.utils import format_number
from invariant_widgets import OutputTextCtrl
# Dimensions related to chart
RECTANGLE_WIDTH = 400.0
RECTANGLE_HEIGHT = 20
# Invariant panel size
_BOX_WIDTH = 76

# scale to use for a bar of value zero
RECTANGLE_SCALE = 0.0001
DEFAULT_QSTAR = 1.0

if sys.platform.count("win32") > 0:
    _STATICBOX_WIDTH = 450
    PANEL_WIDTH = 500
    PANEL_HEIGHT = 430
    FONT_VARIANT = 0
else:
    _STATICBOX_WIDTH = 480
    PANEL_WIDTH = 530
    PANEL_HEIGHT = 430
    FONT_VARIANT = 1

ERROR_COLOR = wx.Colour(255, 0, 0, 128)
EXTRAPOLATION_COLOR = wx.Colour(169, 169, 168, 128)
INVARIANT_COLOR = wx.Colour(67, 208, 128, 128)


[docs]class InvariantContainer(wx.Object): """ This class stores some values resulting resulting from invariant calculations. Given the value of total invariant, this class can also determine the percentage of invariants resulting from extrapolation. """ def __init__(self): # invariant at low range self.qstar_low = None # invariant at low range error self.qstar_low_err = None # invariant self.qstar = None # invariant error self.qstar_err = None # invariant at high range self.qstar_high = None # invariant at high range error self.qstar_high_err = None # invariant total self.qstar_total = None # invariant error self.qstar_total_err = None # scale self.qstar_low_percent = None self.qstar_high_percent = None self.qstar_percent = None # warning message self.existing_warning = False self.warning_msg = "No Details on calculations available...\n"
[docs] def compute_percentage(self): """ Compute percentage of each invariant """ if self.qstar_total is None: self.qstar_percent = None self.qstar_low = None self.qstar_high = None self.check_values() return # compute invariant percentage if self.qstar is None: self.qstar_percent = None else: try: self.qstar_percent = float(self.qstar) / float(self.qstar_total) except: self.qstar_percent = 'error' # compute low q invariant percentage if self.qstar_low is None: self.qstar_low_percent = None else: try: self.qstar_low_percent = float(self.qstar_low)\ / float(self.qstar_total) except: self.qstar_low_percent = 'error' # compute high q invariant percentage if self.qstar_high is None: self.qstar_high_percent = None else: try: self.qstar_high_percent = float(self.qstar_high)\ / float(self.qstar_total) except: self.qstar_high_percent = 'error' wx.CallAfter(self.check_values)
[docs] def check_values(self): """ check the validity if invariant """ if self.qstar_total is None and self.qstar is None: self.warning_msg = "Invariant not calculated.\n" return if self.qstar_total == 0: self.existing_warning = True self.warning_msg = "Invariant is zero. \n" self.warning_msg += "The calculations are likely " self.warning_msg += "to be unreliable!\n" return # warning to the user when the extrapolated invariant is greater than %5 msg = '' if self.qstar_percent == 'error': try: float(self.qstar) except: self.existing_warning = True msg += 'Error occurred when computing invariant from data.\n ' if self.qstar_percent > 1: self.existing_warning = True msg += "Invariant Q contribution is greater " msg += "than 100% .\n" if self.qstar_low_percent == 'error': try: float(self.qstar_low) except: self.existing_warning = True msg += "Error occurred when computing extrapolated invariant" msg += " at low-Q region.\n" elif self.qstar_low_percent is not None: if self.qstar_low_percent >= 0.05: self.existing_warning = True msg += "Extrapolated contribution at Low Q is higher " msg += "than 5% of the invariant.\n" elif self.qstar_low_percent < 0: self.existing_warning = True msg += "Extrapolated contribution at Low Q < 0.\n" elif self.qstar_low_percent > 1: self.existing_warning = True msg += "Extrapolated contribution at Low Q is greater " msg += "than 100% .\n" if self.qstar_high_percent == 'error': try: float(self.qstar_high) except: self.existing_warning = True msg += 'Error occurred when computing extrapolated' msg += ' invariant at high-Q region.\n' elif self.qstar_high_percent is not None: if self.qstar_high_percent >= 0.05: self.existing_warning = True msg += "Extrapolated contribution at High Q is higher " msg += "than 5% of the invariant.\n" elif self.qstar_high_percent < 0: self.existing_warning = True msg += "Extrapolated contribution at High Q < 0.\n" elif self.qstar_high_percent > 1: self.existing_warning = True msg += "Extrapolated contribution at High Q is greater " msg += "than 100% .\n" if (self.qstar_low_percent not in [None, "error"]) and \ (self.qstar_high_percent not in [None, "error"])\ and self.qstar_low_percent + self.qstar_high_percent >= 0.05: self.existing_warning = True msg += "The sum of all extrapolated contributions is higher " msg += "than 5% of the invariant.\n" if self.existing_warning: self.warning_msg = '' self.warning_msg += msg self.warning_msg += "The calculations are likely to be" self.warning_msg += " unreliable!\n" else: self.warning_msg = "No Details on calculations available...\n"
[docs]class InvariantDetailsPanel(wx.Dialog): """ This panel describes proportion of invariants """ def __init__(self, parent=None, id=-1, qstar_container=None, title="Invariant Details", size=(PANEL_WIDTH, PANEL_HEIGHT)): wx.Dialog.__init__(self, parent=parent, id=id, title=title, size=size) # Font size self.SetWindowVariant(variant=FONT_VARIANT) self.parent = parent # self.qstar_container self.qstar_container = qstar_container # warning message self.warning_msg = self.qstar_container.warning_msg # Define scale of each bar self.low_inv_percent = self.qstar_container.qstar_low_percent self.low_scale = self.get_scale(percentage=self.low_inv_percent, scale_name="Extrapolated at Low Q") self.inv_percent = self.qstar_container.qstar_percent self.inv_scale = self.get_scale(percentage=self.inv_percent, scale_name="Inv in Q range") self.high_inv_percent = self.qstar_container.qstar_high_percent self.high_scale = self.get_scale(percentage=self.high_inv_percent, scale_name="Extrapolated at High Q") # Default color the extrapolation bar is grey self.extrapolation_color_low = EXTRAPOLATION_COLOR self.extrapolation_color_high = EXTRAPOLATION_COLOR self.invariant_color = INVARIANT_COLOR # change color of high and low bar when necessary self.set_color_bar() # draw the panel itself self._do_layout() self.set_values() def _define_structure(self): """ Define main sizers needed for this panel """ # Box sizers must be defined first before defining buttons/textctrls # (MAC). self.main_sizer = wx.BoxSizer(wx.VERTICAL) # Sizer related to chart chart_box = wx.StaticBox(self, -1, "Invariant Chart") self.chart_sizer = wx.StaticBoxSizer(chart_box, wx.VERTICAL) self.chart_sizer.SetMinSize((PANEL_WIDTH - 50, 110)) # Sizer related to invariant values self.invariant_sizer = wx.GridBagSizer(4, 4) invariant_box = wx.StaticBox(self, -1, "Numerical Values") self.invariant_box_sizer = wx.StaticBoxSizer(invariant_box, wx.HORIZONTAL) self.invariant_box_sizer.SetMinSize((PANEL_WIDTH - 50, -1)) # Sizer related to warning message warning_box = wx.StaticBox(self, -1, "Warning") self.warning_sizer = wx.StaticBoxSizer(warning_box, wx.VERTICAL) self.warning_sizer.SetMinSize((PANEL_WIDTH - 50, -1)) # Sizer related to button self.button_sizer = wx.BoxSizer(wx.HORIZONTAL) def _layout_shart(self): """ Draw widgets related to chart """ self.panel_chart = wx.Panel(self) self.panel_chart.Bind(wx.EVT_PAINT, self.on_paint) self.chart_sizer.Add(self.panel_chart, 1, wx.EXPAND | wx.ALL, 0) def _layout_invariant(self): """ Draw widgets related to invariant """ uncertainty = "+/-" unit_invariant = '[1/(cm * A^3)]' invariant_txt = wx.StaticText(self, -1, 'Q* from Data ') invariant_txt.SetToolTipString("Invariant in the data set's Q range.") self.invariant_tcl = OutputTextCtrl(self, -1, size=(_BOX_WIDTH, -1)) hint_msg = "Invariant in the data set's Q range." self.invariant_tcl.SetToolTipString(hint_msg) self.invariant_err_tcl = OutputTextCtrl(self, -1, size=(_BOX_WIDTH, -1)) hint_msg = "Uncertainty on the invariant from data's range." self.invariant_err_tcl.SetToolTipString(hint_msg) invariant_units_txt = wx.StaticText(self, -1, unit_invariant) hint_msg = "Unit of the invariant from data's Q range" invariant_units_txt.SetToolTipString(hint_msg) invariant_low_txt = wx.StaticText(self, -1, 'Q* from Low-Q') hint_msg = "Extrapolated invariant from low-Q range." invariant_low_txt.SetToolTipString(hint_msg) self.invariant_low_tcl = OutputTextCtrl(self, -1, size=(_BOX_WIDTH, -1)) hint_msg = "Extrapolated invariant from low-Q range." self.invariant_low_tcl.SetToolTipString(hint_msg) self.invariant_low_err_tcl = OutputTextCtrl(self, -1, size=(_BOX_WIDTH, -1)) hint_msg = "Uncertainty on the invariant from low-Q range." self.invariant_low_err_tcl.SetToolTipString(hint_msg) invariant_low_units_txt = wx.StaticText(self, -1, unit_invariant) hint_msg = "Unit of the extrapolated invariant from low-Q range" invariant_low_units_txt.SetToolTipString(hint_msg) invariant_high_txt = wx.StaticText(self, -1, 'Q* from High-Q') hint_msg = "Extrapolated invariant from high-Q range" invariant_high_txt.SetToolTipString(hint_msg) self.invariant_high_tcl = OutputTextCtrl(self, -1, size=(_BOX_WIDTH, -1)) hint_msg = "Extrapolated invariant from high-Q range" self.invariant_high_tcl.SetToolTipString(hint_msg) self.invariant_high_err_tcl = OutputTextCtrl(self, -1, size=(_BOX_WIDTH, -1)) hint_msg = "Uncertainty on the invariant from high-Q range." self.invariant_high_err_tcl.SetToolTipString(hint_msg) invariant_high_units_txt = wx.StaticText(self, -1, unit_invariant) hint_msg = "Unit of the extrapolated invariant from high-Q range" invariant_high_units_txt.SetToolTipString(hint_msg) # Invariant low iy = 0 ix = 0 self.invariant_sizer.Add(invariant_low_txt, (iy, ix), (1, 1), wx.LEFT | wx.EXPAND | wx.ADJUST_MINSIZE, 15) ix += 1 self.invariant_sizer.Add(self.invariant_low_tcl, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) ix += 1 self.invariant_sizer.Add(wx.StaticText(self, -1, uncertainty), (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) ix += 1 self.invariant_sizer.Add(self.invariant_low_err_tcl, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) ix += 1 self.invariant_sizer.Add(invariant_low_units_txt, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) # Invariant iy += 1 ix = 0 self.invariant_sizer.Add(invariant_txt, (iy, ix), (1, 1), wx.LEFT | wx.EXPAND | wx.ADJUST_MINSIZE, 15) ix += 1 self.invariant_sizer.Add(self.invariant_tcl, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) ix += 1 self.invariant_sizer.Add(wx.StaticText(self, -1, uncertainty), (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) ix += 1 self.invariant_sizer.Add(self.invariant_err_tcl, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) ix += 1 self.invariant_sizer.Add(invariant_units_txt, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) # Invariant high iy += 1 ix = 0 self.invariant_sizer.Add(invariant_high_txt, (iy, ix), (1, 1), wx.LEFT | wx.EXPAND | wx.ADJUST_MINSIZE, 15) ix += 1 self.invariant_sizer.Add(self.invariant_high_tcl, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) ix += 1 self.invariant_sizer.Add(wx.StaticText(self, -1, uncertainty), (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) ix += 1 self.invariant_sizer.Add(self.invariant_high_err_tcl, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) ix += 1 self.invariant_sizer.Add(invariant_high_units_txt, (iy, ix), (1, 1), wx.EXPAND | wx.ADJUST_MINSIZE, 0) self.invariant_box_sizer.Add(self.invariant_sizer, 0, wx.TOP | wx.BOTTOM, 10) def _layout_warning(self): """ Draw widgets related to warning """ # Warning [string] self.warning_msg_txt = wx.StaticText(self, -1, self.warning_msg) if self.qstar_container.existing_warning: self.warning_msg_txt.SetForegroundColour('red') self.warning_sizer.AddMany([(self.warning_msg_txt, 0, wx.LEFT | wx.EXPAND | wx.ADJUST_MINSIZE, 10)]) def _layout_button(self): """ Draw widgets related to button """ # Close button id = wx.NewId() button_ok = wx.Button(self, id, "Ok") button_ok.SetToolTipString("Give Details on Computation") self.Bind(wx.EVT_BUTTON, self.on_close, id=id) self.button_sizer.AddMany([((20, 20), 0, wx.LEFT, 350), (button_ok, 0, wx.RIGHT, 10)]) def _do_layout(self): """ Draw window content """ self._define_structure() self._layout_shart() self._layout_invariant() self._layout_warning() self._layout_button() self.main_sizer.AddMany([(self.chart_sizer, 0, wx.ALL, 10), (self.invariant_box_sizer, 0, wx.ALL, 10), (self.warning_sizer, 0, wx.ALL, 10), (self.button_sizer, 0, wx.ALL, 10)]) self.SetSizer(self.main_sizer)
[docs] def set_values(self): """ Set value of txtcrtl """ value = format_number(self.qstar_container.qstar) self.invariant_tcl.SetValue(value) value = format_number(self.qstar_container.qstar_err) self.invariant_err_tcl.SetValue(value) value = format_number(self.qstar_container.qstar_low) self.invariant_low_tcl.SetValue(value) value = format_number(self.qstar_container.qstar_low_err) self.invariant_low_err_tcl.SetValue(value) value = format_number(self.qstar_container.qstar_high) self.invariant_high_tcl.SetValue(value) value = format_number(self.qstar_container.qstar_high_err) self.invariant_high_err_tcl.SetValue(value)
[docs] def get_scale(self, percentage, scale_name='scale'): """ Check scale receive in this panel. """ scale = RECTANGLE_SCALE try: if percentage in [None, 0.0, "error"]: scale = RECTANGLE_SCALE return scale elif percentage < 0: scale = RECTANGLE_SCALE return scale scale = float(percentage) except: scale = RECTANGLE_SCALE self.warning_msg += "Recieve an invalid scale for %s\n" self.warning_msg += "check this value : %s\n" % str(percentage) return scale
[docs] def set_color_bar(self): """ Change the color for low and high bar when necessary """ self.extrapolation_color_low = EXTRAPOLATION_COLOR self.extrapolation_color_high = EXTRAPOLATION_COLOR self.invariant_color = INVARIANT_COLOR # warning to the user when the extrapolated invariant is greater than %5 if self.low_scale >= 0.05 or self.low_scale > 1 or self.low_scale < 0: self.extrapolation_color_low = ERROR_COLOR if self.high_scale >= 0.05 or self.high_scale > 1 or self.high_scale < 0: self.extrapolation_color_high = ERROR_COLOR if self.inv_scale > 1 or self.inv_scale < 0: self.invariant_color = ERROR_COLOR
[docs] def on_close(self, event): """ Close the current window """ self.Close()
[docs] def on_paint(self, event): """ Draw the chart """ dc = wx.PaintDC(self.panel_chart) try: gc = wx.GraphicsContext.Create(dc) except NotImplementedError: msg = "This build of wxPython does not support " msg += "the wx.GraphicsContext family of classes." dc.DrawText(msg, 25, 25) return # Start the drawing font = wx.SystemSettings.GetFont(wx.SYS_DEFAULT_GUI_FONT) font.SetWeight(wx.BOLD) gc.SetFont(font) # Draw a rectangle path = gc.CreatePath() path.AddRectangle(-RECTANGLE_WIDTH / 2, -RECTANGLE_HEIGHT / 2, RECTANGLE_WIDTH / 2, RECTANGLE_HEIGHT / 2) x_origine = 20 y_origine = 15 # Draw low rectangle gc.PushState() label = "Q* from Low-Q " PathFunc = gc.DrawPath w, h = gc.GetTextExtent(label) gc.DrawText(label, x_origine, y_origine) # Translate the rectangle x_center = x_origine + RECTANGLE_WIDTH * self.low_scale / 2 + w + 10 y_center = y_origine + h gc.Translate(x_center, y_center) gc.SetPen(wx.Pen("black", 1)) gc.SetBrush(wx.Brush(self.extrapolation_color_low)) if self.low_inv_percent is None: low_percent = 'Not Computed' elif self.low_inv_percent == 'error': low_percent = 'Error' else: low_percent = format_number(self.low_inv_percent * 100) + '%' x_center = 20 y_center = -h gc.DrawText(low_percent, x_center, y_center) # Increase width by self.low_scale gc.Scale(self.low_scale, 1.0) PathFunc(path) gc.PopState() # Draw rectangle for invariant gc.PushState() # save it again y_origine += 20 gc.DrawText("Q* from Data ", x_origine, y_origine) # offset to the lower part of the window x_center = x_origine + RECTANGLE_WIDTH * self.inv_scale / 2 + w + 10 y_center = y_origine + h gc.Translate(x_center, y_center) # 128 == half transparent gc.SetBrush(wx.Brush(self.invariant_color)) # Increase width by self.inv_scale if self.inv_percent is None: inv_percent = 'Not Computed' elif self.inv_percent == 'error': inv_percent = 'Error' else: inv_percent = format_number(self.inv_percent * 100) + '%' x_center = 20 y_center = -h gc.DrawText(inv_percent, x_center, y_center) gc.Scale(self.inv_scale, 1.0) gc.DrawPath(path) gc.PopState() # restore saved state # Draw rectangle for high invariant gc.PushState() y_origine += 20 gc.DrawText("Q* from High-Q ", x_origine, y_origine) # define the position of the new rectangle x_center = x_origine + RECTANGLE_WIDTH * self.high_scale / 2 + w + 10 y_center = y_origine + h gc.Translate(x_center, y_center) gc.SetBrush(wx.Brush(self.extrapolation_color_high)) # increase scale by self.high_scale if self.high_inv_percent is None: high_percent = 'Not Computed' elif self.high_inv_percent == 'error': high_percent = 'Error' else: high_percent = format_number(self.high_inv_percent * 100) + '%' x_center = 20 y_center = -h gc.DrawText(high_percent, x_center, y_center) gc.Scale(self.high_scale, 1.0) gc.DrawPath(path) gc.PopState()