Начало
Здравствуйте, в предыдущей статье я показывал как сделать игру на python, а сейчас мы посмотри как сделать простой калькулятор на python tkinter.
Создаём окно 485 на 550. Размеры не важны, мне понравились такие. Так же указываем, что окно не будет изменяться.
from tkinter import *
class Main(Frame):
def __init__(self, root):
super(Main, self).__init__(root)
self.build()
def build(self):
pass
def logicalc(self, operation):
pass
def update():
pass
if __name__ == '__main__':
root = Tk()
root["bg"] = "#000"
root.geometry("485x550+200+200")
root.title("Калькулятор")
root.resizable(False, False)
app = Main(root)
app.pack()
root.mainloop()
Отлично, идём дальше.
Делаем кнопочки
В методе build создаём такой список:
btns = [
"C", "DEL", "*", "=",
"1", "2", "3", "/",
"4", "5", "6", "+",
"7", "8", "9", "-",
"+/-", "0", "%", "X^2"
]
Он отвечает за все кнопки, отображающиеся у нас в окне.
Мы создали список, теперь проходимся циклом и отображаем эти кнопки. Для этого в том же методе пишем следующее:
x = 10
y = 140
for bt in btns:
com = lambda x=bt: self.logicalc(x)
Button(text=bt, bg="#FFF",
font=("Times New Roman", 15),
command=com).place(x=x, y=y,
width=115,
height=79)
x += 117
if x > 400:
x = 10
y += 81
Замечательно, у нас есть кнопочки. Добавляем надпись с выводом результата. Я хочу что бы текст был слева, следовательно, аттрибутов выравнивания текста писать не нужно.
self.formula = "0"
self.lbl = Label(text=self.formula, font=("Times New Roman", 21, "bold"),
bg="#000", foreground="#FFF")
self.lbl.place(x=11, y=50)
Пишем логику
def logicalc(self, operation):
if operation == "C":
self.formula = ""
elif operation == "DEL":
self.formula = self.formula[0:-1]
elif operation == "X^2":
self.formula = str((eval(self.formula))**2)
elif operation == "=":
self.formula = str(eval(self.formula))
else:
if self.formula == "0":
self.formula = ""
self.formula += operation
self.update()
def update(self):
if self.formula == "":
self.formula = "0"
self.lbl.configure(text=self.formula)
Так, как у нас нет ввода с клавиатуры, мы можем позволить себе сделать так, просто проверить на спец. кнопки (C, DEL, =) и в остальных случаях просто добавить это к формуле.
У этого калькулятора множество недочетов, но мы и не стремились сделать его идеальным.
Прошу прощения за ошибки в статье. Пишите, я исправлюсь.
Полный код моей версии калькулятора:
from tkinter import *
class Main(Frame):
def __init__(self, root):
super(Main, self).__init__(root)
self.build()
def build(self):
self.formula = "0"
self.lbl = Label(text=self.formula, font=("Times New Roman", 21, "bold"), bg="#000", foreground="#FFF")
self.lbl.place(x=11, y=50)
btns = [
"C", "DEL", "*", "=",
"1", "2", "3", "/",
"4", "5", "6", "+",
"7", "8", "9", "-",
"(", "0", ")", "X^2"
]
x = 10
y = 140
for bt in btns:
com = lambda x=bt: self.logicalc(x)
Button(text=bt, bg="#FFF",
font=("Times New Roman", 15),
command=com).place(x=x, y=y,
width=115,
height=79)
x += 117
if x > 400:
x = 10
y += 81
def logicalc(self, operation):
if operation == "C":
self.formula = ""
elif operation == "DEL":
self.formula = self.formula[0:-1]
elif operation == "X^2":
self.formula = str((eval(self.formula))**2)
elif operation == "=":
self.formula = str(eval(self.formula))
else:
if self.formula == "0":
self.formula = ""
self.formula += operation
self.update()
def update(self):
if self.formula == "":
self.formula = "0"
self.lbl.configure(text=self.formula)
if __name__ == '__main__':
root = Tk()
root["bg"] = "#000"
root.geometry("485x550+200+200")
root.title("Калькулятор")
root.resizable(False, False)
app = Main(root)
app.pack()
root.mainloop()
Только зарегистрированные пользователи могут участвовать в опросе. Войдите, пожалуйста.
Как бы вы оценили грамотность написания данной статьи?
Проголосовали 167 пользователей.
Воздержались 79 пользователей.
20 декабря, 2016 12:13 пп
106 222 views
| 10 комментариев
Python
Язык программирования Python является отличным инструментом для обработки чисел и математических выражений. На основе этого качества можно создавать полезные программы.
В данном руководстве вам предлагается полезное упражнение: попробуйте написать простую программу командной строки для выполнения вычислений. Итак, в данном руководстве вы научитесь создавать простейший калькулятор в Python 3.
В руководстве используются математические операторы, переменные, условные выражения, функции.
Требования
Для выполнения руководства нужно установить Python 3 на локальную машину и развернуть среду разработки. Все необходимые инструкции можно найти здесь:
- Настройка локальной среды разработки для Python 3 в CentOS 7
- Настройка локальной среды разработки для Python 3 в Windows 10
- Настройка локальной среды разработки для Python 3 в Mac OS X
- Настройка локальной среды разработки для Python 3 в Ubuntu 16.04
1: Строка ввода
Для начала нужно написать строку ввода, с помощью которой пользователи смогут вводить данные для вычислений в калькуляторе.
Для этого используйте встроенную функцию input(), которая принимает сгенерированный пользователем ввод с клавиатуры. В круглых скобках функции input() можно передать строку. Пользовательскому вводу нужно присвоить переменную.
В данной программе пользователь сможет вводить два числа. Запрашивая ввод, нужно добавить пробел в конце строки, чтобы отделить ввод пользователя от строки программы.
number_1 = input('Enter your first number: ')
number_2 = input('Enter your second number: ')
Прежде чем запустить программу, сохраните файл. К примеру, назовём программу calculator.py. теперь можно запустить программу в окне терминала в среде разработки с помощью команды:
python calculator.py
Программа предложит вам ввести два числа:
Enter your first number: 5
Enter your second number: 7
На данный момент калькулятор принимает любые входные данные, не ограничиваясь числами: слова, символы, пробелы, даже enter. Это происходит потому, что функция input() принимает данные как строки и не знает, что в данном случае нужны только числа.
Чтобы программа могла выполнять математические вычисления, она не должна принимать никаких данных, кроме чисел.
В зависимости от предназначения калькулятора, программа может преобразовывать строки функции input() в целые числа или в числа с плавающей точкой. В данном случае целые числа подходят больше. Функцию input() нужно передать внутри функции int(), чтобы преобразовать ввод в целое число.
Читайте также:
Типы данных в Python 3
Преобразование типов данных в Python 3
number_1 = int(input('Enter your first number: '))
number_2 = int(input('Enter your second number: '))
Теперь попробуйте ввести два целых числа:
Enter your first number: 23
Enter your second number: 674
Все работает без ошибок. Однако если вы введёте символы, пробелы или буквы, программа вернёт ошибку:
Enter your first number: hello
Traceback (most recent call last):
File "testing.py", line 1, in <module>
number_1 = int(input('Enter your first number: '))
ValueError: invalid literal for int() with base 10: 'hello'
Итак, вы написали строку для ввода данных в программу.
Примечание: Попробуйте самостоятельно преобразовать входные данные в числа с плавающей точкой.
2: Добавление операторов
Теперь нужно добавить четыре базовых оператора: + (сложение), – (вычитание), * (умножение) и / (деление).
Программу лучше разрабатывать постепенно, чтобы иметь возможность протестировать её на каждом этапе.
Сначала добавьте оператор сложения. Поместите два числа в print, чтобы калькулятор отображал результат.
number_1 = int(input('Enter your first number: '))
number_2 = int(input('Enter your second number: '))
print(number_1 + number_2)
Запустите программу и попробуйте сложить два числа:
Enter your first number: 8
Enter your second number: 3
11
Теперь можно немного усложнить программу. Пусть кроме результата калькулятор также отображает числа, введенные пользователем.
Читайте также: Форматирование строк в Python 3
number_1 = int(input('Enter your first number: '))
number_2 = int(input('Enter your second number: '))
print('{} + {} = '.format(number_1, number_2))
print(number_1 + number_2)
Снова запустите программу и попробуйте ввести какие-нибудь числа:
Enter your first number: 90
Enter your second number: 717
90 + 717 =
807
Теперь пользователь сможет убедиться, что ввел правильные числа.
На данном этапе можно добавить остальные операторы, используя такой же формат:
number_1 = int(input('Enter your first number: '))
number_2 = int(input('Enter your second number: '))
# Addition
print('{} + {} = '.format(number_1, number_2))
print(number_1 + number_2)
# Subtraction
print('{} - {} = '.format(number_1, number_2))
print(number_1 - number_2)
# Multiplication
print('{} * {} = '.format(number_1, number_2))
print(number_1 * number_2)
# Division
print('{} / {} = '.format(number_1, number_2))
print(number_1 / number_2)
Теперь калькулятор может выполнять математические вычисления при помощи операторов +, -, * и /. Далее нужно ограничить количество операций, которые программа может выполнить за один раз.
3: Добавление условного оператора
Читайте также: Условные операторы в Python 3
Добавьте в начало программы calculator.py небольшое описание с перечнем доступных операций. Выбрав один из операторов, пользователь сообщит программе, что именно ей нужно будет делать.
'''
Please type in the math operation you would like to complete:
+ for addition
- for subtraction
* for multiplication
/ for division
'''
Примечание: На самом деле здесь можно использовать любые символы (например, 1 для сложения, b для вычитания и так далее).
Передайте строку внутри функции input() и присвойте переменную значению ввода (к примеру, это будет переменная operation).
operation = input('''
Please type in the math operation you would like to complete:
+ for addition
- for subtraction
* for multiplication
/ for division
''')
number_1 = int(input('Enter your first number: '))
number_2 = int(input('Enter your second number: '))
print('{} + {} = '.format(number_1, number_2))
print(number_1 + number_2)
print('{} - {} = '.format(number_1, number_2))
print(number_1 - number_2)
print('{} * {} = '.format(number_1, number_2))
print(number_1 * number_2)
print('{} / {} = '.format(number_1, number_2))
print(number_1 / number_2)
В эту строку пользователь может ввести любой из предложенных символов, но ничего не произойдёт. Чтобы программа работала, нужно добавить условный оператор. Оператор if будет отвечать за сложение, три оператора elif – за остальные операции; оператор else будет возвращать ошибку, если вместо предложенных операторов пользователь ввёл другой символ.
operation = input('''
Please type in the math operation you would like to complete:
+ for addition
- for subtraction
* for multiplication
/ for division
''')
number_1 = int(input('Enter your first number: '))
number_2 = int(input('Enter your second number: '))
if operation == '+':
print('{} + {} = '.format(number_1, number_2))
print(number_1 + number_2)
elif operation == '-':
print('{} - {} = '.format(number_1, number_2))
print(number_1 - number_2)
elif operation == '*':
print('{} * {} = '.format(number_1, number_2))
print(number_1 * number_2)
elif operation == '/':
print('{} / {} = '.format(number_1, number_2))
print(number_1 / number_2)
else:
print('You have not typed a valid operator, please run the program again.')
Итак, сначала программа предлагает пользователю ввести символ операции. Затем она запрашивает два числа. После этого она отображает пользовательский ввод и результат вычислений. Например, пользователь вводит *, затем 58 и 40.
Please type in the math operation you would like to complete:
+ for addition
- for subtraction
* for multiplication
/ for division
*
Please enter the first number: 58
Please enter the second number: 40
58 * 40 =
2320
Если же на первый запрос программы пользователь введёт символ %, он получит ошибку.
На данный момент программа выполняет все необходимые вычисления. Однако чтобы выполнить другую операцию, программу придётся перезапустить.
4: Определение функций
Чтобы программу не пришлось перезапускать после каждого обработанного примера, нужно определить несколько функций. Для начала поместим весь существующий код в функцию calculate() и добавим в программу ещё один слой. Чтобы программа запускалась, нужно добавить функцию в конец файла.
# Определение функции
def calculate():
operation = input('''
Please type in the math operation you would like to complete:
+ for addition
- for subtraction
* for multiplication
/ for division
''')
number_1 = int(input('Please enter the first number: '))
number_2 = int(input('Please enter the second number: '))
if operation == '+':
print('{} + {} = '.format(number_1, number_2))
print(number_1 + number_2)
elif operation == '-':
print('{} - {} = '.format(number_1, number_2))
print(number_1 - number_2)
elif operation == '*':
print('{} * {} = '.format(number_1, number_2))
print(number_1 * number_2)
elif operation == '/':
print('{} / {} = '.format(number_1, number_2))
print(number_1 / number_2)
else:
print('You have not typed a valid operator, please run the program again.')
# Вызов функции calculate() вне функции
calculate()
Создайте ещё одну функцию, состоящую из условных операторов. Этот блок кода позволит пользователю выбрать: продолжить работу с программой или завершить её. В данном случае операторов будет три: один if, один elif и один else для обработки ошибок.
Пусть функция называется again(). Добавьте её в конец блока def calculate():
...
# Определение функции again()
def again():
# Ввод пользователя
calc_again = input('''
Do you want to calculate again?
Please type Y for YES or N for NO.
''')
# Если пользователь вводит Y, программа запускает функцию calculate()
if calc_again == 'Y':
calculate()
# Если пользователь вводит N, программа попрощается и завершит работу
elif calc_again == 'N':
print('See you later.')
# Если пользователь вводит другой символ, программа снова запускает функцию again()
else:
again()
# Вызов calculate()
calculate()
Также можно устранить чувствительность к регистру: буквы y и n должны восприниматься так же, как Y и N. Для этого добавьте функцию строки str.upper():
...
def again():
calc_again = input('''
Do you want to calculate again?
Please type Y for YES or N for NO.
''')
# Accept 'y' or 'Y' by adding str.upper()
if calc_again.upper() == 'Y':
calculate()
# Accept 'n' or 'N' by adding str.upper()
elif calc_again.upper() == 'N':
print('See you later.')
else:
again()
...
Читайте также: Методы строк в Python 3
Теперь нужно добавить функцию again() в конец функции calculate(), чтобы программа запускала код, который спрашивает пользователя, хочет ли он продолжить работу.
def calculate():
operation = input('''
Please type in the math operation you would like to complete:
+ for addition
- for subtraction
* for multiplication
/ for division
''')
number_1 = int(input('Please enter the first number: '))
number_2 = int(input('Please enter the second number: '))
if operation == '+':
print('{} + {} = '.format(number_1, number_2))
print(number_1 + number_2)
elif operation == '-':
print('{} - {} = '.format(number_1, number_2))
print(number_1 - number_2)
elif operation == '*':
print('{} * {} = '.format(number_1, number_2))
print(number_1 * number_2)
elif operation == '/':
print('{} / {} = '.format(number_1, number_2))
print(number_1 / number_2)
else:
print('You have not typed a valid operator, please run the program again.')
# Добавление функции again() в calculate()
again()
def again():
calc_again = input('''
Do you want to calculate again?
Please type Y for YES or N for NO.
''')
if calc_again.upper() == 'Y':
calculate()
elif calc_again.upper() == 'N':
print('See you later.')
else:
again()
calculate()
Запустите программу в терминале с помощью команды:
python calculator.py
Теперь программу не нужно перезапускать.
5: Дополнительные действия
Написанная вами программа полностью готова к работе. Однако есть ещё много дополнений, которые при желании можно внести в код. Например, вы можете написать приветственное сообщение и добавить его в начало кода:
def welcome():
print('''
Welcome to Calculator
''')
...
# Затем нужно вызвать функции
welcome()
calculate()
Также можно добавить в программу больше функций для обработки ошибок. К примеру, программа должна продолжать работу даже если пользователь вводит слово вместо числа. На данный момент это не так: программа выдаст пользователю ошибку и прекратит работу.
Кроме того, если при выборе оператора деления (/) пользователь выбирает знаменатель 0, он должен получить ошибку:
ZeroDivisionError: division by zero
Для этого нужно написать исключение с помощью оператора try … except.
Программа ограничена 4 операторами, но вы можете расширить этот список:
...
operation = input('''
Please type in the math operation you would like to complete:
+ for addition
- for subtraction
* for multiplication
/ for division
** for power
% for modulo
''')
...
# Для возведения в степень и обработки модуля нужно добавить в код условные операторы.
Также в программу можно добавить операторы цикла.
Существует много способов для настройки обработки ошибок и улучшения каждого проекта. При этом всегда важно помнить, что не существует единственно правильного способа решить ту или иную проблему.
Заключение
Теперь вы знаете, как написать простейший калькулятор. После выполнения руководства вы можете самостоятельно добавить новые функции программы.
Tags: Python, Python 3
В этой статье мы разберём самый простой калькулятор на языке программирования Python, тут стоит сказать, что эта статья подойдёт совсем новичкам, так как сделаем обычный консольный калькулятор, профессиональным или просто опытным она не нужна.
Ещё можете посмотреть статью «Парсер страниц на Python», тоже очень полезна новичкам.
Консольный калькулятор на Python:
Как говорилось выше, мы сделаем легкий калькулятор на Python, и для этого нам нужно создать только один Python файл, я его назову «main.py».
Внутри него создадим функцию которая будет за всё отвечать, вот примерно так:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 |
# Создание функции def main(): # Выводим сообщение print(«Это простой калькулятор на Python») # Запускаем бесконечный цикл while True: # Выводим сообщение какие действия есть print(«Выберите действие которое хотите сделать:n» «Сложить: +n» «Вычесть: -n» «Умножить: *n» «Поделить: /n» «Выйти: qn») # Переменная для хранения действия action = input(«Действие: «) # Если action равен q то if action == «q»: # Выводим сообщение print(«Выход из программы») # Выходим из цикла break # Если action равен +, -, *, /, то if action in (‘+’, ‘-‘, ‘*’, ‘/’): # Присваиваем значение переменной x x = float(input(«x = «)) # Присваиваем значение переменной y y = float(input(«y = «)) # Если action равен + то if action == ‘+’: # Выводим сумму x и y print(‘%.2f + %.2f = %.2f’ % (x, y, x+y)) # Если action равен — то elif action == ‘-‘: # Выводим разность x и y print(‘%.2f — %.2f = %.2f’ % (x, y, x—y)) # Если action равен * то elif action == ‘*’: # Выводим результат умножения x на y print(‘%.2f * %.2f = %.2f’ % (x, y, x*y)) # Если action равен / то elif action == ‘/’: # Если y не равен нулю то if y != 0: # Выводим результат деления x на y print(‘%.2f / %.2f = %.2f’ % (x, y, x/y)) else: # Иначе # Выводим сообщение, что на ноль делить нельзя print(«Деление на ноль!») |
То есть мы создаём функцию где происходит вся логика, внутри неё первым делом выводим сообщение, что мы открыли калькулятор.
Потом запускаем бесконечный цикл и в нём даём выбрать действие, тут как обычно, то есть, если выбираем знак плюса, то будем складывать и т.д..
После идёт проверка команд, если выбрать кнопку «q», то выходим из программы, если же это арифметический знак, то вводим два числа, кладём их в переменные в формате числа с плавающей точкой.
Потом проверяем конкретный знак, зависимо от него и делаем действии, выводим на экран, самое интересное это с делением, мы делаем проверку, если делитель, то есть «y», равен нулю то тогда не будем делить.
Последние что осталось это объявить переменную, вот:
if __name__ == ‘__main__’: main() |
Теперь запускаем файл, для этого переходим в папку и используем эту команду:
Должно быть что-то типа этого:
Как видите наш консольный калькулятор на Python готов, в целом всё очень просто.
Вывод:
В этой статье вы прочитали как сделать самый простой калькулятор на Python, но как вы заметили, статья для новичков, думаю вам понравилась.
Ещё вы можете скачать калькулятор который здесь мы сделали, и сами его разобрать и дополнить.
Подписываетесь на соц-сети:
Оценка:
Загрузка…
Также рекомендую:
Even though web and mobile applications appear to have taken over the software development market, there’s still demand for traditional graphical user interface (GUI) desktop applications. If you’re interested in building these kinds of applications in Python, then you’ll find a wide variety of libraries to choose from. They include Tkinter, wxPython, PyQt, PySide, and a few others.
In this tutorial, you’ll learn the basics of building GUI desktop applications with Python and PyQt.
In this tutorial, you’ll learn how to:
- Create graphical user interfaces with Python and PyQt
- Connect the user’s events on the app’s GUI with the app’s logic
- Organize a PyQt app using a proper project layout
- Create a fully functional GUI application with PyQt
For this tutorial, you’ll create a calculator app with Python and PyQt. This short project will help you grasp the fundamentals and get you up and running with this GUI library.
You can download the source code for the project and all examples in this tutorial by clicking on the link below:
Getting to Know PyQt
PyQt is a Python binding for Qt, which is a set of C++ libraries and development tools providing platform-independent abstractions for graphical user interfaces (GUIs). Qt also provides tools for networking, threads, regular expressions, SQL databases, SVG, OpenGL, XML, and many other powerful features.
Developed by RiverBank Computing Ltd, PyQt’s latest editions are:
- PyQt5: An edition that’s built against Qt 5.x only
- PyQt6: An edition that’s built against Qt 6.x only
In this tutorial, you’ll use PyQt6, as this version is the future of the library. From now on, be sure to consider any mention of PyQt as a reference to PyQt6.
PyQt6 is based on Qt v6. Therefore, it provides classes and tools for GUI creation, XML handling, network communication, regular expressions, threads, SQL databases, web browsing, and other technologies available in Qt. PyQt6 implements binding for many Qt classes in a set of Python modules, which are organized in a top-level Python package called PyQt6
. For PyQt6 to work, you need Python 3.6.1 or later.
PyQt6 is compatible with Windows, Unix, Linux, macOS, iOS, and Android. This is an attractive feature if you’re looking for a GUI framework to develop multiplatform applications that have a native look and feel on each platform.
PyQt6 is available under two licenses:
- The Riverbank Commercial License
- The General Public License (GPL), version 3
Your PyQt6 license must be compatible with your Qt license. If you use the GPL license, then your code must also use a GPL-compatible license. If you want to use PyQt6 to create commercial applications, then you need a commercial license for your installation.
If you need more information about PyQt6 licensing, then check out the license FAQs page on the project’s official documentation.
Installing PyQt
You have several options for installing PyQt on your system or development environment. The recommended option is to use to use binary wheels. Wheels are the standard way to install Python packages from the Python package index, PyPI.
In any case, you need to consider that wheels for PyQt6 are only available for Python 3.6.1 and later. There are wheels for Linux, macOS, and Windows (64-bit).
All of these wheels include copies of the corresponding Qt libraries, so you won’t need to install them separately.
Another installation option is to build PyQt from source. This can be a bit complicated, so you might want to avoid it if possible. If you really need to build from source, then check out what the library’s documentation recommends in those cases.
Alternatively, you have the option of using package managers, such as APT on Linux or Homebrew on macOS, to install PyQt6. In the next few sections, you’ll go through some of the options for installing PyQt6 from different sources and on different platforms.
Virtual Environment Installation With pip
Most of the time, you should create a Python virtual environment to install PyQt6 in an isolated way. To create a virtual environment and install PyQt6 in it, run the following on your command line:
- Windows
- Linux + macOS
PS> python -m venv venv
PS> venvScriptsactivate
(venv) PS> python -m pip install pyqt6
$ python -m venv venv
$ source venv/bin/activate
(venv) $ python -m pip install pyqt6
Here, you first create a virtual environment using the venv
module from the standard library. Then you activate it, and finally you install PyQt6 in it using pip
. Note that you must have Python 3.6.1 or later for the install command to work correctly.
System-Wide Installation With pip
You’ll rarely need to install PyQt directly on your system Python environment. If you ever need to do this kind of installation, then run the following command on your command line or in your terminal window without activating any virtual environment:
$ python -m pip install pyqt6
With this command, you’ll install PyQt6 in your system Python environment directly. You can start using the library immediately after the installation finishes. Depending on your operating system, you may need root or administrator privileges for this installation to work.
Even though this is a fast way to install PyQt6 and start using it right away, it’s not the recommended approach. The recommended approach is to use a Python virtual environment, as you learned in the previous section.
Platform-Specific Installation
Several Linux distributions include binary packages for PyQt6 in their repositories. If this your case, then you can install the library using the distribution’s package manager. On Ubuntu, for example, you can use the following command:
$ sudo apt install python3-pyqt6
With this command, you’ll install PyQt6 and all of its dependencies in your base system, so you can use the library in any of your GUI projects. Note that root privileges are needed, which you invoke here with the sudo
command.
If you’re a macOS user, then you can install PyQt6 using the Homebrew package manager. To do this, open a terminal and run the following command:
After running this command, you’ll have PyQt6 installed on your Homebrew Python environment, and it’ll be ready for you to use.
If you use a package manager on Linux or macOS, then there’s a chance you won’t get the latest version of PyQt6. A pip
installation will be better if you want to ensure that you have the latest release.
Creating Your First PyQt Application
Now that you have a working PyQt installation, you’re ready to create your first GUI app. You’ll create a Hello, World!
application with Python and PyQt. Here are the steps that you’ll follow:
- Import
QApplication
and all the required widgets fromPyQt6.QtWidgets
. - Create an instance of
QApplication
. - Create your application’s GUI.
- Show your application’s GUI.
- Run your application’s event loop, or main loop.
You can download the source code for the examples that you’ll code in this section by clicking the link below:
To kick things off, start by creating a new file called hello.py
in your current working directory:
# hello.py
"""Simple Hello, World example with PyQt6."""
import sys
# 1. Import QApplication and all the required widgets
from PyQt6.QtWidgets import QApplication, QLabel, QWidget
First, you import sys
, which will allow you to handle the application’s termination and exit status through the exit()
function. Then you import QApplication
, QLabel
, and QWidget
from QtWidgets
, which is part of the PyQt6
package. With these imports, you’re done with step one.
To complete step two, you just need to create an instance of QApplication
. Do this as you would create an instance of any Python class:
# hello.py
# ...
# 2. Create an instance of QApplication
app = QApplication([])
In this line of code, you create the instance of QApplication
. You should create your app
instance before you create any GUI object in PyQt.
Internally, the QApplication
class deals with command-line arguments. That’s why you need to pass in a list of command-line arguments to the class constructor. In this example, you use an empty list because your app won’t be handling any command-line arguments.
Step three involves creating the application’s GUI. In this example, your GUI will be based on the QWidget
class, which is the base class of all user interface objects in PyQt.
Here’s how you can create the app’s GUI:
# hello.py
# ...
# 3. Create your application's GUI
window = QWidget()
window.setWindowTitle("PyQt App")
window.setGeometry(100, 100, 280, 80)
helloMsg = QLabel("<h1>Hello, World!</h1>", parent=window)
helloMsg.move(60, 15)
In this code, window
is an instance of QWidget
, which provides all the features that you’ll need to create the application’s window, or form. As its names suggests, .setWindowTitle()
sets the window’s title in your application. In this example, the app’s window will show PyQt App
as its title.
You can use .setGeometry()
to define the window’s size and screen position. The first two arguments are the x
and y
screen coordinates where the window will be placed. The third and fourth arguments are the window’s width
and height
.
Every GUI application needs widgets, or graphical components that make the app’s GUI. In this example, you use a QLabel
widget, helloMsg
, to show the message Hello, World!
on your application’s window.
QLabel
objects can display HTML-formatted text, so you can use the HTML element "<h1>Hello, World!</h1>"
to provide the desired text as an h1
header. Finally, you use .move()
to place helloMsg
at the coordinates (60, 15)
on the application’s window.
You’re done with step three, so you can continue with the final two steps and get your PyQt GUI application ready to run:
# hello.py
# ...
# 4. Show your application's GUI
window.show()
# 5. Run your application's event loop
sys.exit(app.exec())
In this code snippet, you call .show()
on window
. The call to .show()
schedules a paint event, which is a request to paint the widgets that compose a GUI. This event is then added to the application’s event queue. You’ll learn more about PyQt’s event loop in a later section.
Finally, you start the application’s event loop by calling .exec()
. The call to .exec()
is wrapped in a call to sys.exit()
, which allows you to cleanly exit Python and release memory resources when the application terminates.
You can run your first PyQt app with the following command:
When you run this script, you’ll see a window that’ll look something like this:
Your application shows a window based on QWidget
. The window displays the Hello, World!
message. To show the message, it uses a QLabel
widget. And with that, you’ve written your first GUI desktop application using PyQt and Python! Isn’t that cool?
Considering Code Styles
If you check the code of your sample GUI application from the previous section, then you’ll notice that PyQt’s API doesn’t follow PEP 8 coding style and naming conventions. PyQt is built around Qt, which is written in C++ and uses the camel case naming style for functions, methods, and variables. That said, when you start writing a PyQt project, you need to decide which naming style you’ll use.
In this regard, PEP 8 states that:
New modules and packages (including third party frameworks) should be written to these standards, but where an existing library has a different style, internal consistency is preferred. (Source)
In addition, the Zen of Python says:
…practicality beats purity. (Source)
If you want to write consistent PyQt-related code, then you should stick to the framework’s coding style. In this tutorial, you’ll follow the PyQt coding style for consistency. You’ll use camel case instead of the usual Python snake case.
Learning the Basics of PyQt
You’ll need to master the basic components of PyQt if you want to proficiently use this library to develop your GUI applications. Some of these components include:
- Widgets
- Layout managers
- Dialogs
- Main windows
- Applications
- Event loops
- Signals and slots
These elements are the building blocks of any PyQt GUI application. Most of them are represented as Python classes that live in the PyQt6.QtWidgets
module. These elements are extremely important. You’ll learn more about them in the following few sections.
Widgets
Widgets are rectangular graphical components that you can place on your application’s windows to build the GUI. Widgets have several attributes and methods that allow you to tweak their appearance and behavior. They can also paint a representation of themselves on the screen.
Widgets also detect mouse clicks, keypresses, and other events from the user, the window system, and other sources. Each time a widget catches an event, it emits a signal to announce its state change. PyQt has a rich and modern collection of widgets. Each of those widgets serves a different purpose.
Some of the most common and useful PyQt widgets are:
- Buttons
- Labels
- Line edits
- Combo boxes
- Radio buttons
First up is the button. You can create a button by instantiating QPushButton
, a class that provides a classic command button. Typical buttons are Ok
, Cancel
, Apply
, Yes
, No
, and Close
. Here’s how they look on a Linux system:
Buttons like these are perhaps the most commonly used widgets in any GUI. When someone clicks them, your app commands the computer to perform actions. This is how you can execute computations when a user clicks a button.
Up next are labels, which you can create with QLabel
. Labels let you display useful information as text or images:
You’ll use labels like these to explain how to use your app’s GUI. You can tweak a label’s appearance in several ways. A label can even accept HTML-formatted text, as you saw earlier. You can also use labels to specify a keyboard shortcut to move the cursor focus to a given widget on your GUI.
Another common widget is the line edit, also known as the input box. This widget allows you to enter a single line of text. You can create line edits with the QLineEdit
class. Line edits are useful when you need to get the user’s input as plain text.
Here’s how line edits look on a Linux system:
Line edits like these automatically provide basic editing operations like copy, paste, undo, redo, drag, drop, and so on. In the above figure, you can also see that the objects on the first row show placeholder text to inform the user what kind of input is required.
Combo boxes are another fundamental type of widget in GUI applications. You can create them by instantiating QComboBox
. A combo box will present your user with a dropdown list of options in a way that takes up minimal screen space.
Here’s an example of a combo box that provides a dropdown list of popular programming languages:
This combo box is read-only, which means that users can select one of several options but can’t add their own options. Combo boxes can also be editable, allowing users to add new options on the fly. Combo boxes can also contain pixmaps, strings, or both.
The last widget that you’ll learn about is the radio button, which you can create with QRadioButton
. A QRadioButton
object is an option button that you can click to switch on. Radio buttons are useful when you need the user to select one of many options. All options in a radio button are visible on the screen at the same time:
In this radio buttons group, only one button can be checked at a given time. If the user selects another radio button, then the previously selected button will switch off automatically.
PyQt has a large collection of widgets. At the time of this writing, there are over forty available for you to use to create your application’s GUI. Here, you’ve studied only a small sample. However, that’s enough to show you the power and flexibility of PyQt. In the next section, you’ll learn how to lay out different widgets to build modern and fully functional GUIs for your applications.
Layout Managers
Now that you know about widgets and how they’re used to build GUIs, you need to know how to arrange a set of widgets so that your GUI is both coherent and functional. In PyQt, you’ll find a few techniques for laying out the widgets on a form or window. For instance, you can use the .resize()
and .move()
methods to give widgets absolute sizes and positions.
However, this technique can have some drawbacks. You’ll have to:
- Do many manual calculations to determine the correct size and position of every widget
- Do extra calculations to respond to window resize events
- Redo most of your calculations when the window’s layout changes in any way
Another technique involves using .resizeEvent()
to calculate the widget’s size and position dynamically. In this case, you’ll have similar headaches as with the previous technique.
The most effective and recommended technique is to use PyQt’s layout managers. They’ll increase your productivity, mitigate the risk of errors, and improve your code’s maintainability.
Layout managers are classes that allow you to size and position your widgets on the application’s window or form. They automatically adapt to resize events and GUI changes, controlling the size and position of all their child widgets.
PyQt provides four basic layout manager classes:
QHBoxLayout
QVBoxLayout
QGridLayout
QFormLayout
The first layout manager class, QHBoxLayout
, arranges widgets horizontally from left to right, like with the hypothetical widgets in the following figure:
In the horizontal layout, the widgets will appear one next to the other, starting from the left. The code example below shows how to use QHBoxLayout
to arrange three buttons horizontally:
1# h_layout.py
2
3"""Horizontal layout example."""
4
5import sys
6
7from PyQt6.QtWidgets import (
8 QApplication,
9 QHBoxLayout,
10 QPushButton,
11 QWidget,
12)
13
14app = QApplication([])
15window = QWidget()
16window.setWindowTitle("QHBoxLayout")
17
18layout = QHBoxLayout()
19layout.addWidget(QPushButton("Left"))
20layout.addWidget(QPushButton("Center"))
21layout.addWidget(QPushButton("Right"))
22window.setLayout(layout)
23
24window.show()
25sys.exit(app.exec())
Here’s how this example creates a horizontal layout of buttons:
- Line 18 creates a
QHBoxLayout
object calledlayout
. - Lines 19 to 21 add three buttons to
layout
by calling the.addWidget()
method. - Line 22 sets
layout
as your window’s layout with.setLayout()
.
When you run python h_layout.py
from your command line, you’ll get the following output:
The above figure shows three buttons in a horizontal arrangement. The buttons are shown from left to right in the same order as you added them in your code.
The next layout manager class is QVBoxLayout
, which arranges widgets vertically from top to bottom, like in the following figure:
Each new widget will appear beneath the previous one. This layout allows you to to construct vertical layouts and organize your widgets from top to bottom on your GUI.
Here’s how you can create a QVBoxLayout
object containing three buttons:
1# v_layout.py
2
3"""Vertical layout example."""
4
5import sys
6
7from PyQt6.QtWidgets import (
8 QApplication,
9 QPushButton,
10 QVBoxLayout,
11 QWidget,
12)
13
14app = QApplication([])
15window = QWidget()
16window.setWindowTitle("QVBoxLayout")
17
18layout = QVBoxLayout()
19layout.addWidget(QPushButton("Top"))
20layout.addWidget(QPushButton("Center"))
21layout.addWidget(QPushButton("Bottom"))
22window.setLayout(layout)
23
24window.show()
25sys.exit(app.exec())
On line 18, you create an instance of QVBoxLayout
called layout
. In the next three lines, you add three buttons to layout
. Finally, you use the layout
object to arrange the widget in a vertical layout through the .setLayout()
method on line 22.
When you run this sample application, you’ll get a window that looks something like this:
This figure shows three buttons in a vertical arrangement, one below the other. The buttons appear in the same order as you added them to your code, from top to bottom.
The third layout manager in your list is QGridLayout
. This class arranges widgets in a grid of rows and columns. Every widget will have a relative position on the grid. You can define a widget’s position with a pair of coordinates like (row, column)
. Each coordinate must be an integer number. These pairs of coordinates define which cell on the grid a given widget will occupy.
The grid layout will look something like this:
QGridLayout
takes the available space, divides it up into rows
and columns
, and puts each child widget into its own cell.
Here’s how to create a grid layout arrangement in your GUI:
1# g_layout.py
2
3"""Grid layout example."""
4
5import sys
6
7from PyQt6.QtWidgets import (
8 QApplication,
9 QGridLayout,
10 QPushButton,
11 QWidget,
12)
13
14app = QApplication([])
15window = QWidget()
16window.setWindowTitle("QGridLayout")
17
18layout = QGridLayout()
19layout.addWidget(QPushButton("Button (0, 0)"), 0, 0)
20layout.addWidget(QPushButton("Button (0, 1)"), 0, 1)
21layout.addWidget(QPushButton("Button (0, 2)"), 0, 2)
22layout.addWidget(QPushButton("Button (1, 0)"), 1, 0)
23layout.addWidget(QPushButton("Button (1, 1)"), 1, 1)
24layout.addWidget(QPushButton("Button (1, 2)"), 1, 2)
25layout.addWidget(QPushButton("Button (2, 0)"), 2, 0)
26layout.addWidget(
27 QPushButton("Button (2, 1) + 2 Columns Span"), 2, 1, 1, 2
28)
29window.setLayout(layout)
30
31window.show()
32sys.exit(app.exec())
In this example, you create an application that uses a QGridLayout
object to organize its widgets on the screen. Note that, in this case, the second and third arguments that you pass to .addWidget()
are integer numbers defining each widget’s position on the grid.
On lines 26 to 28, you pass two more arguments to .addWidget()
. These arguments are rowSpan
and columnSpan
, and they’re the fourth and fifth arguments passed to the function. You can use them to make a widget occupy more than one row or column, like you did in the example.
If you run this code from your command line, then you’ll get a window that looks something like this:
In this figure, you can see your widgets arranged in a grid of rows and columns. The last widget occupies two columns, as you specified on lines 26 to 28.
The last layout manager that you’ll learn about is QFormLayout
. This class arranges widgets in a two-column layout. The first column usually displays messages in labels. The second column generally contains widgets like QLineEdit
, QComboBox
, QSpinBox
, and so on. These allow the user to enter or edit data regarding the information in the first column.
The following diagram shows how form layouts work in practice:
The left column consists of labels, while the right column consists of input widgets. If you’re developing a database application, then this kind of layout can be a useful tool that’ll increase your productivity when creating input forms.
The following example shows how to create an application that uses a QFormLayout
object to arrange its widgets:
1# f_layout.py
2
3"""Form layout example."""
4
5import sys
6
7from PyQt6.QtWidgets import (
8 QApplication,
9 QFormLayout,
10 QLineEdit,
11 QWidget,
12)
13
14app = QApplication([])
15window = QWidget()
16window.setWindowTitle("QFormLayout")
17
18layout = QFormLayout()
19layout.addRow("Name:", QLineEdit())
20layout.addRow("Age:", QLineEdit())
21layout.addRow("Job:", QLineEdit())
22layout.addRow("Hobbies:", QLineEdit())
23window.setLayout(layout)
24
25window.show()
26sys.exit(app.exec())
Lines 18 to 23 do the hard work in this example. QFormLayout
has a convenient method called .addRow()
. You can use this method to add a two-widget row to the layout. The first argument to .addRow()
should be a label or a string. Then, the second argument can be any widget that allows the user to enter or edit data. In this specific example, you’ve used line edits.
If you run this code, then you’ll get a window that looks something like this:
The above figure shows a window that uses a form layout. The first column contains labels to ask the user for some information. The second column shows widgets that allow the user to enter or edit the required information.
Dialogs
With PyQt, you can develop two types of GUI desktop applications. Depending on the class that you use to create the main form or window, you’ll have one of the following:
- A main window–style application: The application’s main window inherits from
QMainWindow
. - A dialog-style application: The application’s main window inherits from
QDialog
.
You’ll start with dialog-style applications first. In the next section, you’ll learn about main window–style applications.
To develop a dialog-style application, you need to create a GUI class that inherits from QDialog
, which is the base class of all dialog windows. A dialog window is a stand-alone window that you can use as the main window for your application.
A dialog is always an independent window. If a dialog has a parent
, then it’ll display centered on top of the parent widget. Dialogs with a parent will share the parent’s task bar entry. If you don’t set parent
for a given dialog, then the dialog will get its own entry in the system’s task bar.
Here’s an example of how you’d use QDialog
to develop a dialog-style application:
1# dialog.py
2
3"""Dialog-style application."""
4
5import sys
6
7from PyQt6.QtWidgets import (
8 QApplication,
9 QDialog,
10 QDialogButtonBox,
11 QFormLayout,
12 QLineEdit,
13 QVBoxLayout,
14)
15
16class Window(QDialog):
17 def __init__(self):
18 super().__init__(parent=None)
19 self.setWindowTitle("QDialog")
20 dialogLayout = QVBoxLayout()
21 formLayout = QFormLayout()
22 formLayout.addRow("Name:", QLineEdit())
23 formLayout.addRow("Age:", QLineEdit())
24 formLayout.addRow("Job:", QLineEdit())
25 formLayout.addRow("Hobbies:", QLineEdit())
26 dialogLayout.addLayout(formLayout)
27 buttons = QDialogButtonBox()
28 buttons.setStandardButtons(
29 QDialogButtonBox.StandardButton.Cancel
30 | QDialogButtonBox.StandardButton.Ok
31 )
32 dialogLayout.addWidget(buttons)
33 self.setLayout(dialogLayout)
34
35if __name__ == "__main__":
36 app = QApplication([])
37 window = Window()
38 window.show()
39 sys.exit(app.exec())
This application is a bit more elaborate. Here’s what this code does:
- Line 16 defines a
Window
class for the app’s GUI by inheriting fromQDialog
. - Line 18 calls the parent class’s
.__init__()
method usingsuper()
. This call allows you to properly initialize instances of this class. In this example, theparent
argument is set toNone
because this dialog will be your main window. - Line 19 sets the window’s title.
- Line 20 assigns a
QVBoxLayout
object todialogLayout
. - Line 21 assigns a
QFormLayout
object toformLayout
. - Lines 22 to 25 add widgets to
formLayout
. - Line 26 calls
.addLayout()
ondialogLayout
. This call embeds the form layout into the global dialog layout. - Line 27 defines a button box, which provides a convenient space to display the dialog’s buttons.
- Lines 28 to 31 add two standard buttons,
Ok
andCancel
, to the dialog. - Line 32 adds the button box to the dialog by calling
.addWidget()
.
The if __name__ == "__main__":
construct wraps up the app’s main code. This kind of conditional statement is common in Python apps. It ensures that the indented code will only run if the containing file is executed as a program rather than imported as a module. For more about this construct, check out What Does if name == “main” Do in Python?.
The above code example will show a window that looks something like this:
This figure shows the GUI that you’ve created using a QFormLayout
object to arrange the widgets and a QVBoxLayout
layout for the application’s global layout.
Main Windows
Most of the time, your GUI applications will be main window–style apps. This means that they’ll have a menu bar, some toolbars, a status bar, and a central widget that’ll be the GUI’s main element. It’s also common for your apps to have several dialogs to accomplish secondary actions that depend on a user’s input.
You’ll inherit from QMainWindow
to develop main window–style applications. An instance of a class that derives from QMainWindow
is considered the app’s main window and should be unique.
QMainWindow
provides a framework for building your application’s GUI quickly. This class has its own built-in layout, which accepts the following graphical components:
Component | Position on Window | Description |
---|---|---|
One menu bar | Top | Holds the application’s main menu |
One or more toolbars | Sides | Hold tool buttons and other widgets, such as QComboBox , QSpinBox , and more |
One central widget | Center | Holds the window’s central widget, which can be of any type, including a composite widget |
One or more dock widgets | Around the central widget | Are small, movable, and hidable windows |
One status bar | Bottom | Holds the app’s status bar, which shows status information |
You can’t create a main window without a central widget. You need a central widget even if it’s just a placeholder. When this is the case, you can use a QWidget
object as your central widget.
You can set the window’s central widget with the .setCentralWidget()
method. The main window’s layout will allow you to have only one central widget, but it can be a single or a composite widget. The following code example shows you how to use QMainWindow
to create a main window–style application:
1# main_window.py
2
3"""Main window-style application."""
4
5import sys
6
7from PyQt6.QtWidgets import (
8 QApplication,
9 QLabel,
10 QMainWindow,
11 QStatusBar,
12 QToolBar,
13)
14
15class Window(QMainWindow):
16 def __init__(self):
17 super().__init__(parent=None)
18 self.setWindowTitle("QMainWindow")
19 self.setCentralWidget(QLabel("I'm the Central Widget"))
20 self._createMenu()
21 self._createToolBar()
22 self._createStatusBar()
23
24 def _createMenu(self):
25 menu = self.menuBar().addMenu("&Menu")
26 menu.addAction("&Exit", self.close)
27
28 def _createToolBar(self):
29 tools = QToolBar()
30 tools.addAction("Exit", self.close)
31 self.addToolBar(tools)
32
33 def _createStatusBar(self):
34 status = QStatusBar()
35 status.showMessage("I'm the Status Bar")
36 self.setStatusBar(status)
37
38if __name__ == "__main__":
39 app = QApplication([])
40 window = Window()
41 window.show()
42 sys.exit(app.exec())
Here’s how this code works:
- Line 15 creates a class,
Window
, that inherits fromQMainWindow
. - Line 16 defines the class initializer.
- Line 17 calls the base class’s initializer. Again, the
parent
argument is set toNone
because this is your app’s main window, so it must not have a parent. - Line 18 sets the window’s title.
- Line 19 sets a
QLabel
as the window’s central widget. - Lines 20 to 22 call non-public methods to create different GUI elements:
- Lines 24 to 26 create the main menubar with a drop-down menu called Menu. This menu will have a menu option to exit the app.
- Lines 28 to 31 create the toolbar, which will have a toolbar button to exit the app.
- Lines 33 to 36 create the app’s status bar.
When you implement GUI components using their own methods, like you did with the menu bar, toolbar, and status bar in this example, you’re making your code more readable and more maintainable.
When you run the above sample application, you’ll get a window like the following:
As you can confirm, your main window–style application has the following components:
- One main menu generically called Menu
- One toolbar with an Exit tool button
- One central widget consisting of a
QLabel
object with a text message - One status bar at the window’s bottom
That’s it! You’ve learned how to build a main window–style application with Python and PyQt. Up to this point, you’ve learned about some of the more important graphical components in PyQt’s set of widgets. In the next few sections, you’ll study other important concepts related to building GUI applications with PyQt.
Applications
QApplication
is the most foundational class that you’ll use when developing PyQt GUI applications. This class is the core component of any PyQt application. It manages the application’s control flow as well as its main settings.
In PyQt, any instance of QApplication
is an application. Every PyQt GUI application must have one QApplication
instance. Some of the responsibilities of this class include:
- Handling the app’s initialization and finalization
- Providing the event loop and event handling
- Handling most system-wide and application-wide settings
- Providing access to global information, such as the application’s directory, screen size, and so on
- Parsing common command-line arguments
- Defining the application’s look and feel
- Providing localization capabilities
These are just some of the core responsibilities of QApplication
. So, this is a fundamental class when it comes to developing PyQt GUI applications.
One of the most important responsibilities of QApplication
is to provide the event loop and the entire event handling mechanism. In the following section, you’ll take a closer look at what the event loop is and how it works.
Event Loops
GUI applications are event-driven. This means that functions and methods are called in response to user actions, like clicking on a button, selecting an item from a combo box, entering or updating the text in a text edit, pressing a key on the keyboard, and so on. These user actions are commonly known as events.
Events are handled by an event loop, also known as a main loop. An event loop is an infinite loop in which all events from the user, the window system, and any other sources are processed and dispatched. The event loop waits for an event to occur and then dispatches it to perform some task. The event loop continues to work until the application is terminated.
All GUI applications have an event loop. When an event happens, then the loop checks if it’s a terminate event. In that case, the loop finishes, and the application exits. Otherwise, the event is sent to the application’s event queue for further processing, and the loop iterates again. In PyQt6, you can run the app’s event loop by calling .exec()
on the QApplication
object.
For an event to trigger an action, you need to connect the event with the action that you want to execute. In PyQt, you can establish that connection with the signals and slots mechanism, which you’ll explore in the next section.
Signals and Slots
PyQt widgets act as event-catchers. This means that every widget can catch specific events, like mouse clicks, keypresses, and so on. In response to these events, a widget emits a signal, which is a kind of message that announces a change in its state.
The signal on its own doesn’t perform any action. If you want a signal to trigger an action, then you need to connect it to a slot. This is the function or method that’ll perform an action whenever its associated signal is emitted. You can use any Python callable as a slot.
If a signal is connected to a slot, then the slot is called whenever the signal is emitted. If a signal isn’t connected to any slot, then nothing happens and the signal is ignored. Some of the most relevant features of signals and slots include the following:
- A signal can be connected to one or many slots.
- A signal may also be connected to another signal.
- A slot may be connected to one or many signals.
You can use the following syntax to connect a signal and a slot:
widget.signal.connect(slot_function)
This will connect slot_function
to widget.signal
. From now on, whenever .signal
is emitted, slot_function()
will be called.
The code below shows how to use the signals and slots mechanism in a PyQt application:
1# signals_slots.py
2
3"""Signals and slots example."""
4
5import sys
6
7from PyQt6.QtWidgets import (
8 QApplication,
9 QLabel,
10 QPushButton,
11 QVBoxLayout,
12 QWidget,
13)
14
15def greet():
16 if msgLabel.text():
17 msgLabel.setText("")
18 else:
19 msgLabel.setText("Hello, World!")
20
21app = QApplication([])
22window = QWidget()
23window.setWindowTitle("Signals and slots")
24layout = QVBoxLayout()
25
26button = QPushButton("Greet")
27button.clicked.connect(greet)
28
29layout.addWidget(button)
30msgLabel = QLabel("")
31layout.addWidget(msgLabel)
32window.setLayout(layout)
33window.show()
34sys.exit(app.exec())
On line 15, you create greet()
, which you’ll use as a slot. Then in line 27, you connect the button’s .clicked
signal to greet()
. This way, whenever the user clicks the Greet button, the greet()
slot is called and the label object’s text alternates between Hello, World!
and an empty string:
When you click the Greet button, the Hello, World!
message appears and disappears on your application’s main window.
If your slot function needs to receive extra arguments, then you can pass them using functools.partial()
. For example, you can modify greet()
to take an argument, like in the following code:
# signals_slots.py
# ...
def greet(name):
if msg.text():
msg.setText("")
else:
msg.setText(f"Hello, {name}")
# ...
Now greet()
needs to receive an argument called name
. If you want to connect this new version of greet()
to the .clicked
signal, then you can do something like this:
# signals_slots.py
"""Signals and slots example."""
import sys
from functools import partial
# ...
button = QPushButton("Greet")
button.clicked.connect(partial(greet, "World!"))
# ...
For this code to work, you need to import partial()
from functools
first. The call to partial()
returns a function object that behaves similarly to greet()
when called with name="World!"
. Now, when the user clicks on the button, the message Hello, World!
will appear in the label just like before.
The signals and slots mechanism is what you’ll use to give life to your PyQt GUI applications. This mechanism will allow you to turn user events into concrete actions. You can dive deeper into signals and slots by checking out the PyQt6 documentation on the topic.
Now you know the basics of several important concepts of PyQt. With this knowledge and the library’s documentation at hand, you’re ready to start developing your own GUI applications. In the next section, you’ll build your first fully functional GUI application.
Creating a Calculator App With Python and PyQt
In this section, you’ll develop a calculator GUI app using the Model-View-Controller (MVC) design pattern. This pattern has three layers of code, with each one having different roles:
-
The model takes care of your app’s business logic. It contains the core functionality and data. In your calculator app, the model will handle the input values and the calculations.
-
The view implements your app’s GUI. It hosts all the widgets that the end user would need to interact with the application. The view also receives a user’s actions and events. For your example, the view will be the calculator window on your screen.
-
The controller connects the model and the view to make the application work. Users’ events, or requests, are sent to the controller, which puts the model to work. When the model delivers the requested result, or data, in the right format, the controller forwards it to the view. In your calculator app, the controller will receive the target math expressions from the GUI, ask the model to perform calculations, and update the GUI with the result.
Here’s a step-by-step description of how your GUI calculator app will work:
- The user performs an action or request (event) on the view (GUI).
- The view notifies the controller about the user’s action.
- The controller gets the user’s request and queries the model for a response.
- The model processes the controller’s query, performs the required computations, and returns the result.
- The controller receives the model’s response and updates the view accordingly.
- The user finally sees the requested result on the view.
You’ll use this MVC design to build your calculator app with Python and PyQt.
Creating the Skeleton for Your PyQt Calculator App
To kick things off, you’ll start by implementing a minimal skeleton for your application in a file called pycalc.py
. You can get this file and the rest of the source code for your calculator app by clicking the link below:
If you’d prefer to code the project on your own, then go ahead and create pycalc.py
in your current working directory. Open the file in your favorite code editor or IDE and type the following code:
1# pycalc.py
2
3"""PyCalc is a simple calculator built with Python and PyQt."""
4
5import sys
6
7from PyQt6.QtWidgets import QApplication, QMainWindow, QWidget
8
9WINDOW_SIZE = 235
10
11class PyCalcWindow(QMainWindow):
12 """PyCalc's main window (GUI or view)."""
13
14 def __init__(self):
15 super().__init__()
16 self.setWindowTitle("PyCalc")
17 self.setFixedSize(WINDOW_SIZE, WINDOW_SIZE)
18 centralWidget = QWidget(self)
19 self.setCentralWidget(centralWidget)
20
21def main():
22 """PyCalc's main function."""
23 pycalcApp = QApplication([])
24 pycalcWindow = PyCalcWindow()
25 pycalcWindow.show()
26 sys.exit(pycalcApp.exec())
27
28if __name__ == "__main__":
29 main()
This script implements all the boilerplate code that you’ll need to run a basic GUI application. You’ll use this skeleton to build your calculator app.
Here’s how this code works:
-
Line 5 imports
sys
. This module provides theexit()
function, which you’ll use to cleanly terminate the app. -
Line 7 imports the required classes from
PyQt6.QtWidgets
. -
Line 9 creates a Python constant to hold a fixed window size in pixels for your calculator app.
-
Line 11 creates the
PyCalcWindow
class to provide the app’s GUI. Note that this class inherits fromQMainWindow
. -
Line 14 defines the class initializer.
-
Line 15 calls
.__init__()
on the super class for initialization purposes. -
Line 16 sets the window’s title to
"PyCalc"
. -
Line 17 uses
.setFixedSize()
to give the window a fixed size. This ensures that the user won’t be able to resize the window during the app’s execution. -
Lines 18 and 19 create a
QWidget
object and set it as the window’s central widget. This object will be the parent of all the required GUI components in your calculator app. -
Line 21 defines your calculator’s main function. Having a
main()
function like this is a best practice in Python. This function provides the application’s entry point. Insidemain()
, your program does the following:- Line 23 creates a
QApplication
object namedpycalcApp
. - Line 24 creates an instance of the app’s window,
pycalcWindow
. - Line 25 shows the GUI by calling
.show()
on the window object. - Line 26 runs the application’s event loop with
.exec()
.
- Line 23 creates a
Finally, line 29 calls main()
to execute your calculator app. When you run the above script, the following window appears on your screen:
That’s it! You’ve successfully built a fully functional app skeleton for your GUI calculator app. Now you’re ready to continue building the project.
Completing the App’s View
The GUI that you have at this point doesn’t really look like a calculator. You need to finish this GUI by adding a display to show the target math operation and a keyboard of buttons representing numbers and basic math operators. You’ll also add buttons representing other required symbols and actions, like clearing the display.
First, you need to update your imports like in the code below:
# pycalc.py
import sys
from PyQt6.QtCore import Qt
from PyQt6.QtWidgets import (
QApplication,
QGridLayout,
QLineEdit,
QMainWindow,
QPushButton,
QVBoxLayout,
QWidget,
)
# ...
You’ll use a QVBoxLayout
layout manager for the calculator’s global layout. To arrange the buttons, you’ll use a QGridLayout
object. The QLineEdit
class will work as the calculator’s display and QPushButton
will provide the required buttons.
Now you can update the initializer for PyCalcWindow
:
# pycalc.py
# ...
class PyCalcWindow(QMainWindow):
"""PyCalc's main window (GUI or view)."""
def __init__(self):
super().__init__()
self.setWindowTitle("PyCalc")
self.setFixedSize(WINDOW_SIZE, WINDOW_SIZE)
self.generalLayout = QVBoxLayout()
centralWidget = QWidget(self)
centralWidget.setLayout(self.generalLayout)
self.setCentralWidget(centralWidget)
self._createDisplay()
self._createButtons()
# ...
You’ve added the highlighted lines of code. You’ll use .generalLayout
as the app’s general layout. In this layout, you’ll place the display at the top and the keyboard buttons in a grid layout at the bottom.
The calls to ._createDisplay()
and ._createButtons()
won’t work at this point, because you haven’t implemented those methods yet. To fix this issue, you’ll start by coding ._createDisplay()
.
Get back to your code editor and update pycalc.py
like in the following code:
# pycalc.py
# ...
WINDOW_SIZE = 235
DISPLAY_HEIGHT = 35
class PyCalcWindow(QMainWindow):
# ...
def _createDisplay(self):
self.display = QLineEdit()
self.display.setFixedHeight(DISPLAY_HEIGHT)
self.display.setAlignment(Qt.AlignmentFlag.AlignRight)
self.display.setReadOnly(True)
self.generalLayout.addWidget(self.display)
# ...
In this code snippet, you first define a new constant to hold the display height in pixels. Then you define ._createDisplay()
inside PyCalcWindow
.
To create the calculator’s display, you use a QLineEdit
widget. Then you set a fixed height of thirty-five pixels for your display using the DISPLAY_HEIGHT
constant. The display will have its text left-aligned. Finally, the display will be read-only to prevent direct editing by the user. The last line of code adds the display to the calculator’s general layout.
Next up, you’ll implement the ._createButtons()
method to create the required buttons for your calculator’s keyboard. These buttons will live in a grid layout, so you need a way to represent their coordinates on the grid. Each coordinate pair will consist of a row and a column. To represent a coordinate pair, you’ll use a list of lists. Each nested list will represent a row.
Now go ahead and update the pycalc.py
file with the following code:
# pycalc.py
# ...
WINDOW_SIZE = 235
DISPLAY_HEIGHT = 35
BUTTON_SIZE = 40
# ...
In this piece of code, you define a new constant called BUTTON_SIZE
. You’ll use this constant to provide the size of your calculator’s buttons. In this specific example, all the buttons will have a square shape with forty pixels per side.
With this initial setup, you can code the ._createButtons()
method. You’ll use a list of lists to hold the keys or buttons and their position on the calculator keyboard. A QGridLayout
will allow you to arrange the buttons on the calculator’s window:
# pycalc.py
# ...
class PyCalcWindow(QMainWindow):
# ...
def _createButtons(self):
self.buttonMap = {}
buttonsLayout = QGridLayout()
keyBoard = [
["7", "8", "9", "/", "C"],
["4", "5", "6", "*", "("],
["1", "2", "3", "-", ")"],
["0", "00", ".", "+", "="],
]
for row, keys in enumerate(keyBoard):
for col, key in enumerate(keys):
self.buttonMap[key] = QPushButton(key)
self.buttonMap[key].setFixedSize(BUTTON_SIZE, BUTTON_SIZE)
buttonsLayout.addWidget(self.buttonMap[key], row, col)
self.generalLayout.addLayout(buttonsLayout)
# ...
You first create the empty dictionary self.buttonMap
to hold the calculator buttons. Then, you create a list of lists to store the key labels. Each row or nested list will represent a row in the grid layout, while the index of each key label will represent the corresponding column on the layout.
Then you define two for
loops. The outer loop iterates over the rows and the inner loop iterates over the columns. Inside the inner loop, you create the buttons and add them to both self.buttonMap
and buttonsLayout
. Every button will have a fixed size of 40x40
pixels, which you set with .setFixedSize()
and the BUTTON_SIZE
constant.
Finally, you embed the grid layout into the calculator’s general layout by calling .addLayout()
on the .generalLayout
object.
Now your calculator’s GUI will show the display and the buttons gracefully. However, you have no way to update the information shown on the display. You’ll fix this by adding a few extra methods to PyCalcWindow
:
Method | Description |
---|---|
.setDisplayText() |
Sets and updates the display’s text |
.displayText() |
Gets the current display’s text |
.clearDisplay() |
Clears the display’s text |
These methods will provides the GUI’s public interface and complete the view class for your Python calculator app.
Here’s a possible implementation:
# pycalc.py
# ...
class PyCalcWindow(QMainWindow):
# ...
def setDisplayText(self, text):
"""Set the display's text."""
self.display.setText(text)
self.display.setFocus()
def displayText(self):
"""Get the display's text."""
return self.display.text()
def clearDisplay(self):
"""Clear the display."""
self.setDisplayText("")
# ...
Here’s a breakdown of what each method does:
-
.setDisplayText()
uses.setText()
to set and update the display’s text. It also uses.setFocus()
to set the cursor’s focus on the display. -
.displayText()
is a getter method that returns the display’s current text. When the user clicks the equal sign (=
) on the calculator’s keyboard, the app will use the return value of.displayText()
as the math expression to be evaluated. -
.clearDisplay()
sets the display’s text to an empty string (""
) so that the user can introduce a new math expression. This method will be triggered every time the user presses the C button on the calculator’s board.
Now your calculator’s GUI is ready for use! When you run the application, you’ll get a window like the following:
You’ve completed the calculator’s GUI, which looks pretty sleek! However, if you try to do some calculations, then the calculator won’t respond as expected. That’s because you haven’t implemented the model and the controller components. In the next section, you’ll write the calculator’s model.
Implementing the Calculator’s Model
In the MVC pattern, the model is the layer of code that takes care of the business logic. In your calculator app, the business logic is all about basic math calculations. So, your model will evaluate the math expressions that your users introduced in the calculator’s GUI.
The calculator’s model also needs to handle errors. To this end, you’ll define the following global constant:
# pycalc.py
# ...
ERROR_MSG = "ERROR"
WINDOW_SIZE = 235
# ...
This ERROR_MSG
constant is the message that the user will see on the calculator’s display if they introduce an invalid math expression.
With the above change, you’re ready to code your app’s model, which will be a single function in this example:
# pycalc.py
# ...
class PyCalcWindow(QMainWindow):
# ...
def evaluateExpression(expression):
"""Evaluate an expression (Model)."""
try:
result = str(eval(expression, {}, {}))
except Exception:
result = ERROR_MSG
return result
# ...
In evaluateExpression()
, you use eval()
to evaluate a math expression that comes as a string. If the evaluation is successful, then you return result
. Otherwise, you return the predefined error message. Note that this function isn’t perfect. It has a couple of important issues:
- The
try
…except
block doesn’t catch a specific exception, so it’s using a practice that’s discouraged in Python. - The function uses
eval()
, which can lead to some serious security issues.
You’re free to rework the function to make it more reliable and secure. In this tutorial, you’ll use the function as is to keep the focus on implementing the GUI.
Creating the Controller Class for Your Calculator
In this section, you’re going to code the calculator’s controller class. This class will connect the view to the model that you just coded. You’ll use the controller class to make the calculator perform actions in response to user events.
Your controller class needs to perform three main tasks:
- Access the GUI’s public interface.
- Handle the creation of math expressions.
- Connect all the buttons’
.clicked
signals with the appropriate slots.
To perform all these actions, you’ll code a new PyCalc
class in a moment. Go ahead and update pycalc.py
with the following code:
# pytcalc.py
import sys
from functools import partial
# ...
def evaluateExpression(expression):
# ...
class PyCalc:
"""PyCalc's controller class."""
def __init__(self, model, view):
self._evaluate = model
self._view = view
self._connectSignalsAndSlots()
def _calculateResult(self):
result = self._evaluate(expression=self._view.displayText())
self._view.setDisplayText(result)
def _buildExpression(self, subExpression):
if self._view.displayText() == ERROR_MSG:
self._view.clearDisplay()
expression = self._view.displayText() + subExpression
self._view.setDisplayText(expression)
def _connectSignalsAndSlots(self):
for keySymbol, button in self._view.buttonMap.items():
if keySymbol not in {"=", "C"}:
button.clicked.connect(
partial(self._buildExpression, keySymbol)
)
self._view.buttonMap["="].clicked.connect(self._calculateResult)
self._view.display.returnPressed.connect(self._calculateResult)
self._view.buttonMap["C"].clicked.connect(self._view.clearDisplay)
# ...
At the top of pycalc.py
, you import partial()
from functools
. You’ll use this function to connect signals with methods that need to take extra arguments.
Inside PyCalc
, you define the class initializer, which takes two arguments: the app’s model and its view. Then you store these arguments in appropriate instance attributes. Finally, you call ._connectSignalsAndSlots()
to make all the required connections of signals and slots.
In ._calculateResult()
, you use ._evaluate()
to evaluate the math expression that the user has just typed into the calculator’s display. Then you call .setDisplayText()
on the calculator’s view to update the display text with the computation result.
As its name suggests, the ._buildExpression()
method takes care of building the target math expression. To do this, the method concatenates the initial display value with every new value that the user enters on the calculator’s keyboard.
Finally, the ._connectSignalsAndSlots()
method connects all the buttons’ .clicked
signals with the appropriate slots method in the controller class.
That’s it! Your controller class is ready. However, for all this code to work as a real calculator, you need to update the app’s main()
function like in the code below:
# pytcalc.py
# ...
def main():
"""PyCalc's main function."""
pycalcApp = QApplication([])
pycalcWindow = PyCalcWindow()
pycalcWindow.show()
PyCalc(model=evaluateExpression, view=pycalcWindow)
sys.exit(pycalcApp.exec())
This piece of code creates a new instance of PyCalc
. The model
argument to the PyCalc
class constructor holds a reference to the evaluateExpression()
function, while the view
argument holds a reference to the pycalcWindow
object, which provides the app’s GUI. Now your PyQt calculator application is ready to run.
Running the Calculator
Now that you’ve finished writing your calculator app with Python and PyQt, it’s time for a live test! If you run the application from your command line, then you’ll get something like this:
To use PyCalc, enter a valid math expression with your mouse. Then, press Enter or click the equal sign (=
) button to compute and show the expression result on the calculator’s display. That’s it! You’ve developed your first fully functional GUI desktop application with Python and PyQt!
Additional Tools
PyQt6 offers a useful set of additional tools that can help you build solid, modern, and full-featured GUI applications. Some of the most remarkable tools related to PyQt include Qt Designer and the internationalization kit.
Qt Designer allows you to design and build graphical user interfaces using a drag-and-drop interface. You can use this tool to design widgets, dialogs, and main windows by using on-screen forms and a drag-and-drop mechanism. The following animation shows some of Qt Designer’s features:
Qt Designer uses XML .ui
files to store your GUI designs. PyQt includes a module called uic to help with .ui
files. You can also convert the .ui
file content into Python code with a command-line tool called pyuic6
.
PyQt6 also provides a comprehensive set of tools for the internationalization of apps into local languages. The pylupdate6
command-line tool creates and updates translation (.ts
) files, which can contain translations of interface strings. If you prefer GUI tools, then you can use Qt Linguist to create and update .ts
files with translations of interface strings.
Conclusion
Graphical user interface (GUI) applications still hold a substantial share of the software development market. Python offers a handful of frameworks and libraries that can help you develop modern and robust GUI applications.
In this tutorial, you learned how to use PyQt, which is one of the most popular and solid libraries for GUI application development in Python. Now you know how to effectively use PyQt to build modern GUI applications.
In this tutorial, you’ve learned how to:
- Build graphical user interfaces with Python and PyQt
- Connect the user’s events with the app’s logic
- Organize a PyQt application using a proper project layout
- Create a real-world GUI application with PyQt
Now you can use your Python and PyQt knowledge to give life to your own desktop GUI applications. Isn’t that cool?
You can get the source code of your calculator app project and all its associated resources by clicking the link below:
Further Reading
To dive deeper into PyQt and its ecosystem, check out some of the following resources:
- PyQt6’s documentation
- PyQt5’s documentation
- PyQt4’s documentation
- Qt v6’s documentation
- The PyQt wiki
- The Rapid GUI Programming with Python and Qt book
- The Qt Designer manual
- Qt for Python’s documentation
Although the PyQt6 Documentation is the first resource listed here, some important parts of it are still missing or incomplete. Fortunately, you can use the Qt documentation to fill in the blanks.
import
tkinter as tk
import
tkinter.messagebox
from
tkinter.constants
import
SUNKEN
window
=
tk.Tk()
window.title(
'Calculator-GeeksForGeeks'
)
frame
=
tk.Frame(master
=
window, bg
=
"skyblue"
, padx
=
10
)
frame.pack()
entry
=
tk.Entry(master
=
frame, relief
=
SUNKEN, borderwidth
=
3
, width
=
30
)
entry.grid(row
=
0
, column
=
0
, columnspan
=
3
, ipady
=
2
, pady
=
2
)
def
myclick(number):
entry.insert(tk.END, number)
def
equal():
try
:
y
=
str
(
eval
(entry.get()))
entry.delete(
0
, tk.END)
entry.insert(
0
, y)
except
:
tkinter.messagebox.showinfo(
"Error"
,
"Syntax Error"
)
def
clear():
entry.delete(
0
, tk.END)
button_1
=
tk.Button(master
=
frame, text
=
'1'
, padx
=
15
,
pady
=
5
, width
=
3
, command
=
lambda
: myclick(
1
))
button_1.grid(row
=
1
, column
=
0
, pady
=
2
)
button_2
=
tk.Button(master
=
frame, text
=
'2'
, padx
=
15
,
pady
=
5
, width
=
3
, command
=
lambda
: myclick(
2
))
button_2.grid(row
=
1
, column
=
1
, pady
=
2
)
button_3
=
tk.Button(master
=
frame, text
=
'3'
, padx
=
15
,
pady
=
5
, width
=
3
, command
=
lambda
: myclick(
3
))
button_3.grid(row
=
1
, column
=
2
, pady
=
2
)
button_4
=
tk.Button(master
=
frame, text
=
'4'
, padx
=
15
,
pady
=
5
, width
=
3
, command
=
lambda
: myclick(
4
))
button_4.grid(row
=
2
, column
=
0
, pady
=
2
)
button_5
=
tk.Button(master
=
frame, text
=
'5'
, padx
=
15
,
pady
=
5
, width
=
3
, command
=
lambda
: myclick(
5
))
button_5.grid(row
=
2
, column
=
1
, pady
=
2
)
button_6
=
tk.Button(master
=
frame, text
=
'6'
, padx
=
15
,
pady
=
5
, width
=
3
, command
=
lambda
: myclick(
6
))
button_6.grid(row
=
2
, column
=
2
, pady
=
2
)
button_7
=
tk.Button(master
=
frame, text
=
'7'
, padx
=
15
,
pady
=
5
, width
=
3
, command
=
lambda
: myclick(
7
))
button_7.grid(row
=
3
, column
=
0
, pady
=
2
)
button_8
=
tk.Button(master
=
frame, text
=
'8'
, padx
=
15
,
pady
=
5
, width
=
3
, command
=
lambda
: myclick(
8
))
button_8.grid(row
=
3
, column
=
1
, pady
=
2
)
button_9
=
tk.Button(master
=
frame, text
=
'9'
, padx
=
15
,
pady
=
5
, width
=
3
, command
=
lambda
: myclick(
9
))
button_9.grid(row
=
3
, column
=
2
, pady
=
2
)
button_0
=
tk.Button(master
=
frame, text
=
'0'
, padx
=
15
,
pady
=
5
, width
=
3
, command
=
lambda
: myclick(
0
))
button_0.grid(row
=
4
, column
=
1
, pady
=
2
)
button_add
=
tk.Button(master
=
frame, text
=
"+"
, padx
=
15
,
pady
=
5
, width
=
3
, command
=
lambda
: myclick(
'+'
))
button_add.grid(row
=
5
, column
=
0
, pady
=
2
)
button_subtract
=
tk.Button(
master
=
frame, text
=
"-"
, padx
=
15
, pady
=
5
, width
=
3
, command
=
lambda
: myclick(
'-'
))
button_subtract.grid(row
=
5
, column
=
1
, pady
=
2
)
button_multiply
=
tk.Button(
master
=
frame, text
=
"*"
, padx
=
15
, pady
=
5
, width
=
3
, command
=
lambda
: myclick(
'*'
))
button_multiply.grid(row
=
5
, column
=
2
, pady
=
2
)
button_div
=
tk.Button(master
=
frame, text
=
"/"
, padx
=
15
,
pady
=
5
, width
=
3
, command
=
lambda
: myclick(
'/'
))
button_div.grid(row
=
6
, column
=
0
, pady
=
2
)
button_clear
=
tk.Button(master
=
frame, text
=
"clear"
,
padx
=
15
, pady
=
5
, width
=
12
, command
=
clear)
button_clear.grid(row
=
6
, column
=
1
, columnspan
=
2
, pady
=
2
)
button_equal
=
tk.Button(master
=
frame, text
=
"="
, padx
=
15
,
pady
=
5
, width
=
9
, command
=
equal)
button_equal.grid(row
=
7
, column
=
0
, columnspan
=
3
, pady
=
2
)
window.mainloop()
This article will explore how to build a simple GUI calculator using Python and tkinter.
Table of Contents
- Introduction
- Project overview
- How to make a calculator in Python
- Step 1: Build GUI main screen for the calculator
- Step 2: Add the equation line to GUI main screen
- Step 3: Create buttons for the calculator
- Step 4: Add buttons to GUI
- Step 5: Assign actions to buttons
- Complete code to make calculator in Python
- Conclusion
Introduction
If you are reading this article, you probably already know how to make a calculator program in Python, and now you want to learn how to build a GUI (graphic user interface) for such a program, so it’s more presentable and easily reusable.
Throughout this tutorial we will be working with tkinter, which is the standard GUI library for Python. If you are interested in learning more about the library and its functionality, the detailed guide is available online.
Project overview
Before we dive into the code, let’s first discuss what type of calculator we are going to build, its functionality, and its “must haves”.
The GUI we will be building will look like this:
The following features should be considered as “must haves” for this app:
- It must have an equation line (screen) to display the mathematical equation and the result.
- It must have buttons which must represent numerical values or mathematical operations, and be clickable.
- It must have the mathematical operations functionality (addition, subtraction, multiplication, division).
- It must perform mathematical calculations and return the result (=) as well as clear the equation line (c).
How to make a calculator in Python
In the following steps we will go through the process of building a GUI calculator in Python.
Step 1: Build GUI main screen for calculator in Python
As a first step, we are going to import the required dependency: tkinter.
Next we will define a Calculator class and add the __init__ method, and the execution statement. This will be our main frame on top of which we will be adding the GUI and calculator functionality.
This may look like a difficult to understand code, but we will discuss in detail what each part does, so bare with me.
from tkinter import *
#Create a calculator class
class Calculator:
#Step 2 and 3: create the __init__ method
def __init__(self, master):
'''
Method that initializes the object's attributes
'''
#Step 4: Assign reference to the main window of the application
self.master = master
#Step 4: Add a name to our application
master.title("Python Calculator")
#Step 5: Execution
if __name__ == '__main__':
#Step 6: Create the main window of an application
root = Tk()
#Step 6: Tell our calculator class to use this window
my_gui = Calculator(root)
#Step 6: Executable loop on the application, waits for user input
root.mainloop()
Let’s discuss it step by step:
Step 1.1:
Create a Calculator class (which is an individual object with specified behaviour).
Step 1.2:
Define our first method: __init__.
This method is specific to Python classes and acts as a constructor in object oriented concepts. This method is called right away when an object is created and allows it to initialize its attributes.
Step 1.3:
Pass two arguments into __init__ method: self and master:
- self represents the instance of the class, which basically means that __init__ is a method of the Calculator class.
- master is a local variable that we pass into the __init__ method, and what it does is it creates a top level widget (usually the main window of an application). More details will be discussed further.
Step 1.4:
Create a reference to the main window of the application self.master=master within the Calculator class.
Then add a title to our application master.title(“Python Calculator”)
Step 1.5:
Add the standard execution starter if __name__==__’main‘__. This is a standard procedure and you can read find more details here.
Step 1.6:
Specify what should happen when we run the code:
- Define a local variable root and assign Tk() to it. Tk() is a tkinter class which creates the main application window.
- Define another local variable my_gui and assign Calculator class to is while passing root as an argument into it. Essentially this step creates a GUI by telling a class to use root (which is Tk()) as it’s main window.
- Add root.mainloop() where mainloop() is a method of Tk() basically telling the app to wait for events (clicks) and update the UI accordingly.
That was a lot of information to consume at once. But we have completed the first part of the tutorial and now we can actually see our initial screen.
Rune the code and you should see is the following:
Step 2: Add the equation line to GUI main screen
As the next step we would like to add an equation line to our calculator GUI, which is the same thing as the “display” on a real calculator where all of the operations are shown.
This part is very intuitive and requires us to add only two lines of code to the __init__ method:
#Step 1: Create a line where we display the equation
self.equation = Entry(master, width=36, borderwidth=5)
#Step 2: Assign a position for the equation line in the grey application window
self.equation.grid(row=0, column=0, columnspan=4, padx=10, pady=10)
The reason we are adding it to the __init__ method is because we want this to be “created” as soon as the application starts.
Let’s discuss it step by step:
Step 2.1:
We need to create some equation line that we want to add to our Calculator class, hence we define it as self.equation and then use a tkinter class Entry() to create a widget.
There are a few arguments that go into the Entry() class. We need to let it know that it should belong to our master window, as well as some basic styling properties such as width and borderwith (feel free to change these, but for now please follow with the values I have assigned).
Step 2.2:
We will then “place” the equation line onto the main screen. So far we have created the equation line which belongs to the Calculator class. But it doesn’t know yet where on the main screen it should be positioned. We will place at the top of our GUI . To do this, we qilluse the .grid() method of Entry() class along with some parameters.
We already know that it needs to be assigned to our “display”, hence the form self.equation.grid().
Now what goes into .grid() needs a little explanation:
- row – recall that Python indexing starts with 0, so first row has an index of 0 and the first column has an index of 0 as well.
- columnspan – it will specify how many “columns” will this line occupy. Take a look at the final image of the calculator app in the project overview section, an you will notice that the app buttons are placed into 4 columns. Since our equation line takes up the whole width of the screen, it means that it takes place of exactly 4 columns.
Note: this is also why we use the width of 36, since it can be divided by 4 without the remainder and keep the layout unaffected. - padx and pady – the basic paddings that are added to the top, bottom, left, and right of the equation line just to visually give it a little more space.
Making these additions to our code, you should arrive at the following:
from tkinter import *
#Create a calculator class
class Calculator:
def __init__(self, master):
"""
Method that initializes the object's attributes
"""
#Assign reference to the main window of the application
self.master = master
#Add a name to our application
master.title("Python Calculator")
#Create a line where we display the equation
self.equation = Entry(master, width=36, borderwidth=5)
#Assign a position for the equation line in the grey application window
self.equation.grid(row=0, column=0, columnspan=4, padx=10, pady=10)
#Execution
if __name__ == '__main__':
#Create the main window of an application
root = Tk()
#Tell our calculator class to use this window
my_gui = Calculator(root)
#Executable loop on the application, waits for user input
root.mainloop()
Run the code and you should see is the following:
Notice that the application window actually became smaller, since now it is wrapped around the equation display. You can also freely type anything you want inside of the line just to test its ‘entry’ properties.
Step 3: Create buttons for calculator in Python
We have created the calculator main screen and the equation line attached to it. The next step is to add buttons to this GUI so it looks like an actual calculator.
This is the most tedious part of the code and technically a little more advanced than the previous sections. Let’s go through all of the code in detail.
We are going to create a new method createButton() and add the following code to it:
# Step 1: Create a method that belongs to an instance of the Calculator class
def createButton(self):
"""
Method that creates the buttons
INPUT: nothing
OUTPUT: creates a button
"""
#Step 2: We first create each button one by one with the value we want
#Using addButton() method which is described below
b0 = self.addButton(0)
b1 = self.addButton(1)
b2 = self.addButton(2)
b3 = self.addButton(3)
b4 = self.addButton(4)
b5 = self.addButton(5)
b6 = self.addButton(6)
b7 = self.addButton(7)
b8 = self.addButton(8)
b9 = self.addButton(9)
b_add = self.addButton('+')
b_sub = self.addButton('-')
b_mult = self.addButton('*')
b_div = self.addButton('/')
b_clear = self.addButton('c')
b_equal = self.addButton('=')
# Step 3: Arrange the buttons into lists to represent calculator rows
row1 = [b7,b8,b9,b_add]
row2 = [b4,b5,b6,b_sub]
row3 = [b1,b2,b3,b_mult]
row4 = [b_clear,b0,b_equal,b_div]
# Step4: Assign each button to a particular location on the GUI
r = 1
for row in [row1, row2, row3, row4]:
c = 0
for buttn in row:
buttn.grid(row=r, column=c, columnspan=1)
c += 1
r += 1
Let’s discuss it step by step:
Step 3.1:
Define a new method createButton() and pass self as an argument to it. This basically assigns this method to the instance of the Calculator class.
Step 3.2:
Create button objects. You will notice that all of the buttons follow the same template: b[some name]=self.addButton(some value). It is a tedious process to create each button one by one, and it can be automated using a loop. For the purposes of explaining this step in detail, each button will be hard coded. Essentially each button name represents the action that we want this button to take. For example, b9 is a button for 9, b_add is a button for addition, and so on.
Now, the question that comes up is: what is this self.addButton()? It is another method that we will define later in the code. What it will do is it will create the button object we need with the value we need. It will be explained in the next section. The logic here is that for each button inside the createButton() method, we call the addButton() method in order to generate the button.
Step 3.3:
Arrange buttons into rows in which they appear on the calculator. What we want it to have 4 rows with 4 buttons in each row, so they appear in the following order:
Row 1: 7, 8, 9, +
Row 2: 4, 5, 6, –
Row 3: 1, 2, 3, *
Row 4: c, 0, =, /
Step 3.4:
Populate each row with the buttons. As the final step of adding buttons, we will loop through every column of every row and populate it with the buttons.
One thing to keep in mind is that we are starting from row index 1 to 4 (from r=1), since our row index 0 is populated with the equation line. And our columns will be indexed from 0 to 3.
Additional step:
This is an important step, and isn’t the chunk of code displayed above. After we have created this method to create the buttons, we need to “call” it from our __init__ method by adding self.createButton() to it.
After this section, the complete code so far should look like the following:
from tkinter import *
#Create a calculator class
class Calculator:
def __init__(self, master):
"""
Method that initializes the object's attributes
"""
#Assign reference to the main window of the application
self.master = master
#Add a name to our application
master.title("Python Calculator")
#Create a line where we display the equation
self.equation = Entry(master, width=36, borderwidth=5)
#Assign a position for the equation line in the grey application window
self.equation.grid(row=0, column=0, columnspan=4, padx=10, pady=10)
#Execute the .creteButton() method
self.createButton()
def createButton(self):
"""
Method that creates the buttons
INPUT: nothing
OUTPUT: creates a button
"""
#We first create each button one by one with the value we want
#Using addButton() method which is described below
b0 = self.addButton(0)
b1 = self.addButton(1)
b2 = self.addButton(2)
b3 = self.addButton(3)
b4 = self.addButton(4)
b5 = self.addButton(5)
b6 = self.addButton(6)
b7 = self.addButton(7)
b8 = self.addButton(8)
b9 = self.addButton(9)
b_add = self.addButton('+')
b_sub = self.addButton('-')
b_mult = self.addButton('*')
b_div = self.addButton('/')
b_clear = self.addButton('c')
b_equal = self.addButton('=')
#Arrange the buttons into lists which represent calculator rows
row1 = [b7,b8,b9,b_add]
row2 = [b4,b5,b6,b_sub]
row3 = [b1,b2,b3,b_mult]
row4 = [b_clear,b0,b_equal,b_div]
#Assign each button to a particular location on the GUI
r = 1
for row in [row1, row2, row3, row4]:
c = 0
for buttn in row:
buttn.grid(row=r, column=c, columnspan=1)
c += 1
r += 1
#Execution
if __name__=='__main__':
#Create the main window of an application
root = Tk()
#Tell our calculator class to use this window
my_gui = Calculator(root)
#Executable loop on the application, waits for user input
root.mainloop()
Note: if you try to run the code now, you should get an error. The reason for the error is that we used some addButton() method inside our createButton() method, but we haven’t specified what exactly it does. We are going to cover this in the next section.
Step 4: Add buttons to GUI
In this section we will work on adding the created buttons to the GUI.
Remember the addButton() method we used to create each button? This is exactly what we will focus on adding now. This method creates and returns a tkinter Button object that we will then add to the GUI.
The following few lines of code will help us:
# Step 1: Create a method that belongs to an instance of the Calculator class,
# and takes the "value" of the button as the second argument.
def addButton(self,value):
"""
Method to create a button
INPUT: value of the button (1,2,3,4,5,6,7,8,9,0,+,-,*,/,c,=)
OUTPUT: returns a designed button object
"""
#Step 2: Create the button object
return Button(self.master, text=value, width=9)
Let’s discuss it step by step:
Step 4.1:
Define a new method addButton() and pass two arguments to it:
- self assigns this method to the instance of the Calculator class
- value is the actual “value” that we want to display on each of the buttons in our GUI
Step 4.2:
We create a Button class with a few parameters that we need:
- self.master – it assigns the Button object to the Calculator class.
- text=value – it will display the “value” we passed as an argument as text displayed on the button.
- width=9 – it is just styling, so we can adjust the width of the buttons.
Now let’s add it to our main code and run it:
from tkinter import *
#Create a calculator class
class Calculator:
def __init__(self, master):
"""
Method that initializes the object's attributes
"""
#Assign reference to the main window of the application
self.master = master
#Add a name to our application
master.title("Python Calculator")
#Create a line where we display the equation
self.equation = Entry(master, width=36, borderwidth=5)
#Assign a position for the equation line in the grey application window
self.equation.grid(row=0, column=0, columnspan=4, padx=10, pady=10)
#Execute the .creteButton() method
self.createButton()
def createButton(self):
"""
Method to create a button
INPUT: nothing
OUTPUT: creates a button
"""
#We first create each button one by one with the value we want
#Using addButton() method which is described below
b0 = self.addButton(0)
b1 = self.addButton(1)
b2 = self.addButton(2)
b3 = self.addButton(3)
b4 = self.addButton(4)
b5 = self.addButton(5)
b6 = self.addButton(6)
b7 = self.addButton(7)
b8 = self.addButton(8)
b9 = self.addButton(9)
b_add = self.addButton('+')
b_sub = self.addButton('-')
b_mult = self.addButton('*')
b_div = self.addButton('/')
b_clear = self.addButton('c')
b_equal = self.addButton('=')
#Arrange the buttons into lists which represent calculator rows
row1 = [b7,b8,b9,b_add]
row2 = [b4,b5,b6,b_sub]
row3 = [b1,b2,b3,b_mult]
row4 = [b_clear,b0,b_equal,b_div]
#Assign each button to a particular location on the GUI
r = 1
for row in [row1, row2, row3, row4]:
c = 0
for buttn in row:
buttn.grid(row=r, column=c, columnspan=1)
c += 1
r += 1
def addButton(self,value):
"""
Method to process the creation of a button and make it clickable
INPUT: value of the button (1,2,3,4,5,6,7,8,9,0,+,-,*,/,c,=)
OUTPUT: returns a designed button object
"""
return Button(self.master, text=value, width=9)
#Execution
if __name__ == '__main__':
#Create the main window of an application
root = Tk()
#Tell our calculator class to use this window
my_gui = Calculator(root)
#Executable loop on the application, waits for user input
root.mainloop()
Run the code and you should see the following:
It is a full built GUI with clickable buttons. It works like a regular calculator app now, except nothing happens when the buttons are clicked.
Step 5: Assign Actions to Buttons
The last step if to add the actual mathematical functionality for the calculator app.
Let’s look at the following code:
# Step 1: Create a method that belongs to an instance of the Calculator class,
# and takes the "value" of the button as the second argument.
def clickButton(self, value):
"""
Method to add actions for button clicks
INPUT: value of the button (1,2,3,4,5,6,7,8,9,0,+,-,*,/,c,=)
OUTPUT: what action will be performed when a button is clicked
"""
#Step 2: Get the equation that's entered by the user
current_equation = str(self.equation.get())
#Step 3: If user clicked "c", then clear the screen
if value == 'c':
self.equation.delete(-1, END)
#Step 4: If user clicked "=", then compute the answer and display it
elif value == '=':
answer = str(eval(current_equation))
self.equation.delete(-1, END)
self.equation.insert(0, answer)
#Step 5: If user clicked any other button, then add it to the equation line
else:
self.equation.delete(0, END)
self.equation.insert(0, current_equation+value)
This part is where we add the main mathematical calculations and it is a little more complicated than previous parts.
Let’s discuss the steps:
Step 5.1:
Define a new method clickButton() and pass two arguments:
- self assigns this method to the instance of the Calculator class
- value is the actual “value” that we want to display on each of the buttons in our GUI
Step 5.2:
In order for the calculator to know what to do next, it needs to know what is displayed in the equation line. In order to get the content from the equation line, we use the .get() method of the equation line (which is a tkinter Entry class).
This means that if, for example, we have “1+1” in the equation line, this function will take this line and store it as a string in the local current_equation variable.
Step 5.3:
Here we start adding logic for the clicks, and the simplest one to do is what should be the action performed by the calculator when a user clicks “c”.
We would want the equation line cleared if a user clicks “c”, and this is exactly what we do by using .delete() method of the equation line (which is a tkinter Entry class).
Step 5.4:
The second part of our conditioning is to add the action performed by the calculator when a user clicks “=”. What we want to do is to compute an answer and display it in the equation line.
It will include a few steps:
- First thing we do inside the elif part is compute the current_equation which we already have from Step 1. We use the Python eval() method which parses the string expression and “evaluates” it as if you ran it in Python directly and store it as a local variable answer. For example, if your current_equation is “1+1” and is a string, if you run 1+1 in Python, it will give you 2 as output. This is exactly what we do here.
- Then we delete the current text from the equation line using .delete() method of a tkinter Entry class (similar to how we cleared the line when user clicked “c”).
- And finally insert our answer into the equation line using .insert() method of a tkinter Entry class.
Step 5.5:
And the last step in the conditioning is to add what we can call a “catch all” condition that we put in the else part. Essentially, if a user is clicking anything else but “c” or “=”, it means that they are clicking on a digit from 0 to 9 or a mathematical operation sign (+, -, *, /). All we want to do in that case is to add it to our equation line.
– First, we delete the existing text in the equation line .delete() method of a tkinter Entry class.
– Then we concatenate the current_equation with value (which is the recent click) and insert it into the equation line using .insert() method of a tkinter Entry class. As an example, if you currently have “1+” in your equation line and then click “1”, what we do is glue together “1+” and “1” to get “1+1” and paste in back into the equation line.
Additional step:
The last step that we need to do is to let our buttons know what action belongs to each of them when they are clicked. To do that we need to go back to our addButton() method and add one more parameter to the Button() object creation. What we need to add is a command. A command is a parameter for the button to let it know what it should do, and what each button should essentially do is use clickButton() method as its parameter. So we need to change the code in the addButton() method from:
return Button(self.master, text=value, width=9)
To:
return Button(
self.master,
text=value,
width=9,
command=lambda: self.clickButton(str(value)),
)
Now, when our buttons are created, the GUI knows what the assigned action to a click of each button.
Complete code to make calculator in Python
The final code for the complete basic GUI calculator in Python that we just made should look like this:
from tkinter import *
#Create a calculator class
class Calculator:
def __init__(self, master):
"""
Method that initializes the object's attributes
"""
#Assign reference to the main window of the application
self.master = master
#Add a name to our application
master.title("Python Calculator")
#Create a line where we display the equation
self.equation = Entry(master, width=36, borderwidth=5)
#Assign a position for the equation line in the grey application window
self.equation.grid(row=0, column=0, columnspan=4, padx=10, pady=10)
#Execute the .createButton() method
self.createButton()
def createButton(self):
"""
Method to create a button
INPUT: nothing
OUTPUT: creates a button
"""
#We first create each button one by one with the value we want
#Using addButton() method which is described below
b0 = self.addButton(0)
b1 = self.addButton(1)
b2 = self.addButton(2)
b3 = self.addButton(3)
b4 = self.addButton(4)
b5 = self.addButton(5)
b6 = self.addButton(6)
b7 = self.addButton(7)
b8 = self.addButton(8)
b9 = self.addButton(9)
b_add = self.addButton("+")
b_sub = self.addButton("-")
b_mult = self.addButton("*")
b_div = self.addButton("/")
b_clear = self.addButton("c")
b_equal = self.addButton("=")
#Arrange the buttons into lists which represent calculator rows
row1 = [b7, b8, b9, b_add]
row2 = [b4, b5, b6, b_sub]
row3 = [b1, b2, b3, b_mult]
row4 = [b_clear, b0, b_equal, b_div]
#Assign each button to a particular location on the GUI
r = 1
for row in [row1, row2, row3, row4]:
c = 0
for buttn in row:
buttn.grid(row=r, column=c, columnspan=1)
c += 1
r += 1
def addButton(self, value):
"""
Method to process the creation of a button and make it clickable
INPUT: value of the button (1,2,3,4,5,6,7,8,9,0,+,-,*,/,c,=)
OUTPUT: returns a designed button object
"""
return Button(
self.master,
text=value,
width=9,
command=lambda: self.clickButton(str(value)),
)
def clickButton(self, value):
"""
Method to add actions for button clicks
INPUT: value of the button (1,2,3,4,5,6,7,8,9,0,+,-,*,/,c,=)
OUTPUT: what action will be performed when a particular button is clicked
"""
#Get the equation that's entered by the user
current_equation = str(self.equation.get())
#If user clicked "c", then clear the screen
if value == "c":
self.equation.delete(-1, END)
#If user clicked "=", then compute the answer and display it
elif value == "=":
answer = str(eval(current_equation))
self.equation.delete(-1, END)
self.equation.insert(0, answer)
#If user clicked any other button, then add it to the equation line
else:
self.equation.delete(0, END)
self.equation.insert(-1, current_equation + value)
#Execution
if __name__ == "__main__":
#Create the main window of an application
root = Tk()
#Tell our calculator class to use this window
my_gui = Calculator(root)
#Executable loop for the application, waits for user input
root.mainloop()
This code is also available on my GitHub page.
Conclusion
In this article we covered how you can create a custom basic GUI calculator in Python using tkinter library. I also encourage you to check out my other posts on Python Programming.
Feel free to leave comments below if you have any questions or have suggestions for some edits.
In this Python tutorial, we will make a calculator in python, which can add, subtract, divide, and multiply depending upon the user-entered input.
Also, we will discuss how to create a Python Calculator using Tkinter.
Let us see how we can create a calculator in python using functions.
As you can see that we have created a simple calculator in python, which can perform different arithmetical operations like add, subtract, multiply, and divide. The user-defined function is add(), subtract(), multiply() and divide() will evaluate respective operation and display the output.
Example:
def add(x,y):
return x + y
def subtract(x,y):
return x - y
def multiply(x,y):
return x * y
def divide(x,y):
return x / y
print("Select operation.")
print("1.Add")
print("2.Subtract")
print("3.Multiply")
print("4.Divide")
while True:
choice = input("Enter your choice(1/2/3/4): ")
if choice in ('1', '2', '3', '4'):
number1 = float(input("Enter first number: "))
number2 = float(input("Enter second number: "))
if choice == '1':
print(number1, "+", number2, "=", add(number1, number2))
elif choice == '2':
print(number1, "-", number2, "=", subtract(number1, number2))
elif choice == '3':
print(number1, "*", number2, "=", multiply(number1, number2))
elif choice == '4':
print(number1, "/", number2, "=", divide(number1, number2))
break
else:
print("Entered input is invalid")
You can refer to the below screenshot to make a calculator in Python.
Output:
Here, we ask the user to choose an operation of their choice. If the option is 1,2,3, and 4 then it will be valid else it will display “Invalid input”. If the option is valid then we have to enter the two numbers and according to the user entered choice, it will perform the operation.
You may also like Python Tkinter Stopwatch and How to Take User Input and Store in Variable using Python Tkinter.
Calculator in python using class
Let’s see step by step how we can create a class which will perform basic calculator operation.
- To make a calculator using class. Classes are required to create an object. Classes make the code more efficient and simple enough to understand.
- First, we will create a class calculator and all the function is defined like addition, subtraction, multiplication, and division.
- Self is used while calling the function using obj1.function(). Here, the function will call itself.
- Now, we will take the input from the user and the object is created.
- And at last, create the user choice to perform the operation they need and print the value in the output.
Example:
class calculator:
def addition(self):
print(x + y)
def subtraction(self):
print(x - y)
def multiplication(self):
print(x * y)
def division(self):
print(x / y)
x = int(input("Enter first number:"))
y = int(input("Enter second number:"))
obj1 = calculator()
choice = 1
while choice !=0:
print("1. ADDITION")
print("2. SUBTRACTION")
print("3. MULTIPLICATION")
print("4. DIVISION")
choice = int(input("Enter your choice:"))
if choice == 1:
print(obj1.addition())
elif choice == 2:
print(obj1.subtraction())
elif choice == 3:
print(obj1.multiplication())
elif choice == 4:
print(obj1.division())
else:
print("Invalid choice")
Now, you can see code with the output.
Output:
You may like, How to Set Background to be an Image in Python Tkinter
Calculator in python without function
We have created a simple calculator in python without a function that can perform different arithmetical operations like add, subtract, multiply, and divide.
Example:
first_val = input("Enter 1st number: ")
first_val = int(first_val)
cal = input("Enter operator: ")
second_val = input("Enter 2nd number: ")
second_val = int(second_val)
if cal == "+":
print(first_val + second_val)
elif cal == "-":
print(first_val - second_val)
elif cal == "*":
print(first_val * second_val)
elif cal == "/":
print(first_val / second_val)
Below screenshot shows how we can create a calculator without function.
You may also like, How to display data in textboxes using Python Tkinter?
Calculator in python code
In this, we will see the code for a calculator in python. Here, we create an individual function to perform the operation like add, subtract, multiply, and divide. It will accept the user’s input along with the choice of operator and return the result.
Example:
def add(x, y):
return x + y
def subtract(x, y):
return x - y
def multiply(x, y):
return x * y
def divide(x, y):
return x / y
print("select an operation")
print("+")
print("-")
print("*")
print("/")
choice = input("Enter any operator to use")
a = int(input("Enter first number: "))
b = int(input("Enter second number: "))
if choice == '+':
print(a,"+",b,"=", add(a,b))
elif choice == '-':
print(a,"-",b,"=", subtract(a,b))
elif choice == '*':
print(a,"*",b,"=", multiply(a,b))
elif choice == '/':
print(a,"/",b,"=", divide(a,b))
else:
print("Invalid input")
You can see the below screenshot for calculator in python code and the output.
Output:
This is how we can make a calculator in Python.
Read: Python Tkinter Mainloop
Python Tkinter Calculator
Let us see how to create a simple calculator using Python Tkinter. I hope this Python Tkinter calculator example will help you to create a simple calculator in Python.
If you are new to Python GUI programming or Python Tkinter, check out Python GUI Programming (Python Tkinter)
Create a Simple Calculator using Python Tkinter
- Python Tkinter is used for various GUI applications, one of them is calculator.
- Working with calculator app will help you in understanding python better.
- In this section we will provide you entire source code with explanation.
- Please advised to read the entire blog & leave a comment to clarify any doubt.
Planning for Python Tkinter Calculator
In this section, we will discuss the secret of training yourself before writing a code for the python Tkinter calculator. We will discuss the look of an application, functions it is going to perform, and source of inspiration.
- The source of inspiration is the calculator on the Windows 10 operating system.
- It has a grey background in the display & black background on the buttons.
- The text color of the entire application is white.
- The application always opens at the bottom right side.
- Numbers are displayed on the right side of the screen.
After observing the entire windows 10 calculator application, now it’s time to write things we are going to use or implement in our application.
- Every time application will open on the bottom right side.
- the application will have a grey background in the display & black background in the buttons.
- The text color of the entire application will be white
- a window will have a fixed size & can’t be resized.
- The calculator will have the basic functionality of addition, subtraction, multiplication & division.
- Numbers will be displayed on the right side of the screen.
Algorithm to Create Python Tkinter Calculator
In this section, we will discuss the algorithm for creating a python Tkinter calculator. Also, we will provide the data-flow diagram (DFD) for the same.
ws = Tk()
ws.title('Calculator')
ws.geometry('250x400+500+100')
ws.resizable(0,0)
- the first line of the code is used to initialize the parent layer.
- the second line is used to set the title of the window. The title is the name on the top of the window next to the feather.
- the third line is set the width, height & position of the window.
- width = 250
- height = 400
- position x = 500
- position y = 100
- Resizable determines the amount of screen that can be stretched. The screen can be stretched on the x & y-axis. Since we have provided both the values as 0. That means the screen can’t be stretched either way.
frame_1 = Frame(ws)
frame_1.pack(expand=True, fill=BOTH)
- To place widgets like labels & buttons we have to create 4 frames.
- Each frame is allowed to fully expand in the x and y direction if space is available.
- In this way all the frames will have similar space & buttons placed in them will have symmetry.
expand=True
: This allows for expansion in the available space.fill=both
: direction for espansion
key_1 = Button(
frame_1,
text='1',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display(1)
)
In this code, we are creating a button. To understand the button please refer to our button section. Here, under the command, we can call a function but we cannot pass an argument to a function. That is why we have used the anonymous function lambda here. To know more about the anonymous function please refer to our section on the anonymous function.
Here is the DFD of calculator
Global Variables
num = ''
- Variables used in a function are limited to that function only
- if we want to use a particular variable in other function
- then we have to declare that variable as a global variable.
- In this code, we have declared num as a global variable.
- any function referring to this variable must use the global keyword before the variable name
- Changes made by any function on this variables will be observed by other functions too.
- for example, if any function sets the value of the num to ’10’ then other functions using num will also have num value as ’10’.
Explaining function:
equal_btn()
def equal_btn():
global num
add=str(eval(num))
scr_lbl['text'] = add
num=''
- equal_btn() function is used multiple times with the slight change in variable name in line 3 & line 4.
- here, built-in function eval() plays an important role.
- eval() takes a string as input and converts it into an integer & performs the operation.
- that means if the string ‘2+3’ is passed to eval it will return 5 as output
def clear_scr(number)
def clear_scr():
global num
num = ''
scr_lbl['text'] = num
- this function is called when C is clicked by user.
- it clears everything on the display and sets the value of num to empty string.
def display(number)
def display(number):
global num
num = num + str(number)
scr_lbl['text'] = num
- this function is responsible for displaying numbers on the screen.
- every time user clicks on the numbered button, it concat with the existing string.
- so when clicking on 1 it displays 1 on the screen. then when we click on 2 it displays 12 and so on.
Code:
from tkinter import *
ws = Tk()
ws.title('Calculator')
ws.geometry('250x400+500+100')
ws.resizable(0,0)
# global variables
num = ''
# functions
def display(number):
global num
num = num + str(number)
scr_lbl['text'] = num
def clear_scr():
global num
num = ''
scr_lbl['text'] = num
def equal_btn():
global num
add=str(eval(num))
scr_lbl['text'] = add
num=''
def equal_btn():
global num
sub=str(eval(num))
scr_lbl['text'] = sub
num=''
def equal_btn():
global num
mul=str(eval(num))
scr_lbl['text'] = mul
num=''
def equal_btn():
global num
div=str(eval(num))
scr_lbl['text'] = div
num=''
var = StringVar()
# frames
frame_1 = Frame(ws)
frame_1.pack(expand=True, fill=BOTH)
frame_2 = Frame(ws)
frame_2.pack(expand=True, fill=BOTH)
frame_3 = Frame(ws)
frame_3.pack(expand=True, fill=BOTH)
frame_4 = Frame(ws)
frame_4.pack(expand=True, fill=BOTH)
# label
scr_lbl = Label(
frame_1,
textvariable='',
font=('Arial', 20),
anchor = SE,
bg = '#595954',
fg = 'white'
)
scr_lbl.pack(expand=True, fill=BOTH)
# buttons
key_1 = Button(
frame_1,
text='1',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display(1)
)
key_1.pack(expand=True, fill=BOTH, side=LEFT)
key_2 = Button(
frame_1,
text='2',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display(2)
)
key_2.pack(expand=True, fill=BOTH, side=LEFT)
key_3 = Button(
frame_1,
text='3',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display(3)
)
key_3.pack(expand=True, fill=BOTH, side=LEFT)
key_add = Button(
frame_1,
text='+',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display('+')
)
key_add.pack(expand=True, fill=BOTH, side=LEFT)
key_4 = Button(
frame_2,
text='4',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display(4)
)
key_4.pack(expand=True, fill=BOTH, side=LEFT)
key_5 = Button(
frame_2,
text='5',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display(5)
)
key_5.pack(expand=True, fill=BOTH, side=LEFT)
key_6 = Button(
frame_2,
text='6',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display(6)
)
key_6.pack(expand=True, fill=BOTH, side=LEFT)
key_sub = Button(
frame_2,
text='-',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display('-')
)
key_sub.pack(expand=True, fill=BOTH, side=LEFT)
key_7 = Button(
frame_3,
text='7',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display(7)
)
key_7.pack(expand=True, fill=BOTH, side=LEFT)
key_8 = Button(
frame_3,
text='8',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display(8)
)
key_8.pack(expand=True, fill=BOTH, side=LEFT)
key_9 = Button(
frame_3,
text='9',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display(9)
)
key_9.pack(expand=True, fill=BOTH, side=LEFT)
key_mul = Button(
frame_3,
text='*',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display('*')
)
key_mul.pack(expand=True, fill=BOTH, side=LEFT)
key_clr = Button(
frame_4,
text='C',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = clear_scr
)
key_clr.pack(expand=True, fill=BOTH, side=LEFT)
key_0 = Button(
frame_4,
text='0',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display(0)
)
key_0.pack(expand=True, fill=BOTH, side=LEFT)
key_res = Button(
frame_4,
text='=',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = equal_btn
)
key_res.pack(expand=True, fill=BOTH, side=LEFT)
key_div = Button(
frame_4,
text='/',
font=('Arial', 22),
border = 0,
relief = GROOVE,
bg = '#2E2E2B',
fg = 'white',
command = lambda: display('/')
)
key_div.pack(expand=True, fill=BOTH, side=LEFT)
ws.mainloop()
Output:
So in this tutorial we have learned about how to create calculator using python tkinter.
You may like the following Python tutorials:
- Regular Expressions in Python
- Python Booleans
- Python print 2 decimal places
- Python generators (Python Generator vs Iterator)
- Python Recursion
- Python sort list of tuples
- Python invalid literal for int() with base 10
- Remove Unicode characters in python
- Comment lines in Python
- How to make a matrix in Python
- Python Tkinter Progress bar
- Python Tkinter Exit Program
In this Python tutorial, we have learned how to make a calculator in Python. Also, we have covered these below topics:
- Calculator in python using class
- Calculator in python without function
- Calculator in python code
- Python Tkinter calculator
Python is one of the most popular languages in the United States of America. I have been working with Python for a long time and I have expertise in working with various libraries on Tkinter, Pandas, NumPy, Turtle, Django, Matplotlib, Tensorflow, Scipy, Scikit-Learn, etc… I have experience in working with various clients in countries like United States, Canada, United Kingdom, Australia, New Zealand, etc. Check out my profile.