pyqtcalc.py 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. #!/usr/bin/env python
  2. """ Script to create PyQt application """
  3. import sys
  4. from functools import partial
  5. import qdarktheme
  6. from PyQt6 import QtWidgets
  7. from PyQt6.QtCore import Qt
  8. WINDOW_SIZE = 235 # window size in pixels
  9. DISPLAY_HEIGHT = 45 # display height in pixels
  10. BUTTON_SIZE = 40 # button size in pixels
  11. ERROR_MSG = "ERROR" # error message
  12. class PyQtCalc:
  13. """pyqtcalc controller class"""
  14. def __init__(self, model, view):
  15. self._model = model
  16. self._view = view
  17. self._connectSignalAndSlot()
  18. def _calculateResult(self):
  19. """calculate and update display result"""
  20. result = self._model(expression=self._view.displayText())
  21. self._view.setDisplayText(result)
  22. def _buildExpression(self, subExpression):
  23. """build math expression and display result"""
  24. if self._view.displayText() == ERROR_MSG:
  25. self._view.clearDisplay()
  26. expression = self._view.displayText() + subExpression
  27. self._view.setDisplayText(expression)
  28. def _connectSignalAndSlot(self):
  29. """connect button signals with slots method"""
  30. for keySymbol, button in self._view.buttonMap.items():
  31. if keySymbol not in {"=", "C"}:
  32. button.clicked.connect(partial(self._buildExpression, keySymbol))
  33. self._view.buttonMap["="].clicked.connect(self._calculateResult)
  34. self._view.display.returnPressed.connect(self._calculateResult)
  35. self._view.buttonMap["C"].clicked.connect(self._view.clearDisplay)
  36. class MainWindow(QtWidgets.QMainWindow):
  37. """pyqtcalc main window (GUI or view)"""
  38. def __init__(self):
  39. super().__init__()
  40. self.setWindowTitle("PyQt Calculator")
  41. self.setFixedSize(WINDOW_SIZE, WINDOW_SIZE)
  42. self.generalLayout = QtWidgets.QVBoxLayout()
  43. centralWidget = QtWidgets.QWidget(self)
  44. centralWidget.setLayout(self.generalLayout)
  45. self.setCentralWidget(centralWidget)
  46. self._createDisplay()
  47. self._createButtons()
  48. def _createDisplay(self):
  49. """create application's display"""
  50. self.display = QtWidgets.QLineEdit()
  51. self.display.setFixedHeight(DISPLAY_HEIGHT)
  52. self.display.setAlignment(Qt.AlignmentFlag.AlignRight)
  53. self.display.setReadOnly(True)
  54. self.generalLayout.addWidget(self.display)
  55. def _createButtons(self):
  56. """create application's keyboard buttons"""
  57. self.buttonMap = {}
  58. buttonLayout = QtWidgets.QGridLayout()
  59. keyboard = [
  60. ["7", "8", "9", "/", "C"],
  61. ["4", "5", "6", "*", "("],
  62. ["1", "2", "3", "-", ")"],
  63. ["0", ".", "%", "+", "="],
  64. ]
  65. for row, keys in enumerate(keyboard):
  66. for col, key in enumerate(keys):
  67. self.buttonMap[key] = QtWidgets.QPushButton(key)
  68. self.buttonMap[key].setFixedSize(BUTTON_SIZE, BUTTON_SIZE)
  69. buttonLayout.addWidget(self.buttonMap[key], row, col)
  70. self.generalLayout.addLayout(buttonLayout)
  71. def setDisplayText(self, text):
  72. """sets and updates display's text"""
  73. self.display.setText(text)
  74. self.display.setFocus()
  75. def displayText(self):
  76. """gets current display's text"""
  77. return self.display.text()
  78. def clearDisplay(self):
  79. """clears display's text"""
  80. self.setDisplayText("")
  81. def evaluateExpression(expression):
  82. """evaluate math expressions (model)"""
  83. try:
  84. result = str(eval(expression, {}, {}))
  85. except Exception:
  86. result = ERROR_MSG
  87. return result
  88. def main():
  89. """pyqtcalc main function"""
  90. # create an instance of QApplication
  91. app = QtWidgets.QApplication(sys.argv)
  92. app.setStyle("Fusion") # set Fusion system style
  93. qdarktheme.setup_theme() # apply dark theme to application
  94. window = MainWindow() # application's GUI
  95. window.show() # show application's GUI
  96. PyQtCalc(model=evaluateExpression, view=window)
  97. sys.exit(app.exec()) # application's event loop
  98. if __name__ == "__main__":
  99. sys.exit(main())