Evgenii Legotckoi
Evgenii Legotckoi02 квітня 2017 р. 14:22

PyQt5 - Урок 007. Працює з QML QtQuick (Сигнали та слоти)

А тепер глибше зануримося в роботу з Qt за допомогою PyQt5, скориставшись сучасними можливостями Qt. Під такими можливостями я маю на увазі QtQuick і QML. PyQt5 дозволяє використовувати класи Qt, які можуть обробляти код QML, а отже, можна написати інтерфейс на QML, а також передавати сигнали в QML шар і викликати слоти об'єктів , успадкованих від QObject з шару QML.

Щоб познайомитися з такими можливостями PyQt5, напишемо програму, яка реалізує такі завдання:

  • Інтерфейс програми повинен бути написаний на QML
  • Повинний бути реалізований клас, успадкований від QObject і написаний на python, з яким взаємодіятимемо з QML
  • Додаток за допомогою даного класу повинен буде складати та віднімати цілі числа

Зовнішній вигляд програми повинен виглядати так:


Структура проекту

У проекті буде всього два файли:

  • main .py - файл програми на python, там же буде клас для обчислень
  • main.qml - файл інтерфейсу на QML

Сигнали в PyQt5

Сигнатура сигналу в загальному випадку виглядатиме так:

PyQt5.QtCore.pyqtSignal
( types [, name [, revision=0 [, arguments=[] ]]])

Створіть один або кілька перевантажених незв'язаних сигналів як атрибут класу.

| Параметри: | * types – Типи, що визначають сигнатуру C++ сигналу. Кожен тип може бути об'єктом типу Python або рядком, який є іменем типу C++. Альтернативно кожен може бути послідовністю аргументів типу. І тут кожна послідовність визначає сигнатуру перевантаження іншого сигналу. Перше навантаження буде використовуватися за замовчуванням.
name – Назва сигналу. Якщо його опущено, використовується ім'я атрибута класу. Це може бути поставлене лише як аргумент ключового слова.
revision – ревізія сигналу, який експортується до QML. Це може бути поставлене лише як аргумент ключового слова.
* arguments – Послідовність імен аргументів сигналу, які експортуються до QML. Це може бути поставлене лише як аргумент ключового слова.
|

Слоти в PyQt5

Для оголошення слотів PyQt5 використовується спеціальний декоратор.

PyQt5.QtCore.pyqtSlot
( types [, name [, result [, revision=0 ]]])

| Параметри: | * types – Типи, що визначають сигнатуру слота C++. Кожен тип може бути об'єктом типу Python або рядком, який є ім'ям типу C++.
name – Ім'я слота, який відображатиметься на C++. Якщо опущено, буде використано ім'я Python, що прикрашається. Це може бути поставлене лише як аргумент ключового слова.
revision – ревізія слота, який експортується до QML. Це може бути поставлене лише як аргумент ключового слова.
* result – Тип результату і може бути об'єктом типу Python або рядком, що визначає тип C++. Це може бути поставлене лише як аргумент ключового слова.
|

main .py

from PyQt5.QtGui import QGuiApplication
from PyQt5.QtQml import QQmlApplicationEngine
from PyQt5.QtCore import QObject, pyqtSignal, pyqtSlot


class Calculator(QObject):
    def __init__(self):
        QObject.__init__(self)

    # cигнал передающий сумму
    # обязательно даём название аргументу через arguments=['sum']
    # иначе нельзя будет его забрать в QML
    sumResult = pyqtSignal(int, arguments=['sum'])

    subResult = pyqtSignal(int, arguments=['sub'])

    # слот для суммирования двух чисел
    @pyqtSlot(int, int)
    def sum(self, arg1, arg2):
        # складываем два аргумента и испускаем сигнал
        self.sumResult.emit(arg1 + arg2)

    # слот для вычитания двух чисел
    @pyqtSlot(int, int)
    def sub(self, arg1, arg2):
        # вычитаем аргументы и испускаем сигнал
        self.subResult.emit(arg1 - arg2)


if __name__ == "__main__":
    import sys

    # создаём экземпляр приложения
    app = QGuiApplication(sys.argv)
    # создаём QML движок
    engine = QQmlApplicationEngine()
    # создаём объект калькулятора
    calculator = Calculator()
    # и регистрируем его в контексте QML
    engine.rootContext().setContextProperty("calculator", calculator)
    # загружаем файл qml в движок
    engine.load("main.qml")

    engine.quit.connect(app.quit)
    sys.exit(app.exec_())

main.qml

import QtQuick 2.5
import QtQuick.Controls 1.4
import QtQuick.Layouts 1.2

ApplicationWindow {
    visible: true
    width: 640
    height: 240
    title: qsTr("PyQt5 love QML")
    color: "whitesmoke"

    GridLayout {
        anchors.top: parent.top
        anchors.left: parent.left
        anchors.right: parent.right
        anchors.margins: 9

        columns: 4
        rows: 4
        rowSpacing: 10
        columnSpacing: 10

        Text {
            text: qsTr("First number")
        }

        // Поле ввода первого числа
        TextField {
            id: firstNumber
        }

        Text {
            text: qsTr("Second number")
        }

        // Поле ввода второго числа
        TextField {
            id: secondNumber
        }

        Button {
            height: 40
            Layout.fillWidth: true
            text: qsTr("Sum numbers")

            Layout.columnSpan: 2

            onClicked: {
                // Вызываем слот калькулятора, чтобы сложить числа
                calculator.sum(firstNumber.text, secondNumber.text)
            }
        }

        Text {
            text: qsTr("Result")
        }

        // Здесь увидим результат сложения
        Text {
            id: sumResult
        }

        Button {
            height: 40
            Layout.fillWidth: true
            text: qsTr("Subtraction numbers")

            Layout.columnSpan: 2

            onClicked: {
                // Вызываем слот калькулятора, чтобы вычесть числа
                calculator.sub(firstNumber.text, secondNumber.text)
            }
        }

        Text {
            text: qsTr("Result")
        }

        // Здесь увидим результат вычитания
        Text {
            id: subResult
        }
    }

    // Здесь забираем результат сложения или вычитания чисел
    Connections {
        target: calculator

        // Обработчик сигнала сложения
        onSumResult: {
            // sum было задано через arguments=['sum']
            sumResult.text = sum
        }

        // Обработчик сигнала вычитания
        onSubResult: {
            // sub было задано через arguments=['sub']
            subResult.text = sub
        }
    }
}

Рекомендуємо хостинг TIMEWEB
Рекомендуємо хостинг TIMEWEB
Стабільний хостинг, на якому розміщується соціальна мережа EVILEG. Для проектів на Django радимо VDS хостинг.

Вам це подобається? Поділіться в соціальних мережах!

v
  • 03 серпня 2017 р. 17:49

Спасибо. Все очень доходчиво!

P
  • 05 березня 2018 р. 06:32

Подскажите как закидывать компоненты из python на форму qml? Я хочу график matplotlib закинуть на page qml

Evgenii Legotckoi
  • 05 березня 2018 р. 06:52

Вот сейчас Вы меня конкретно загрузили этим вопросом. Не работал с этой библиотекой, но я глянул, что есть сети по этому поводу и кое-какие мысли уже есть у меня. Но надо ещё самому разобраться, что можно придумать. Сегодня/завтра после работы гляну, и потом отвечу Вам, какие мысли у меня будут.

Но вообще, всё скорее всего должно сводиться к множественному наследованию от QQuickPaitedItem и класса библиотек matplotlib.

 
С теоретической исследовательской точки зрения мне это интересно. С практической не уверен, что стоит использовать, тем более, что есть Qt Charts.
 
Но я гляну, что там можно сделать.
P
  • 05 березня 2018 р. 06:58

Спасибо, Вам, за отклик. У меня существует готовое решение с использованием pyqt5 + matplotlib. Где я просто на форму кидаю виджет, а виджет есть график matplotlib. Теперь хотелось бы интерфейс приятный сделать. Вчера потратил весь день на Qt Charts и осознал, что он очень не очень. Ощущение, что студенты на коленках делали.

Evgenii Legotckoi
  • 05 березня 2018 р. 07:06
Ощущение, что студенты на коленках делали.
Ну... может поэтому Qt Charts вынесли из коммерческой лицензии и теперь он доступен в Community Edition ))))
Пожалуй, будет зависеть от того, что вам требуется, возможно, что его функционала действительно не хватает...

Если у вас есть готовый минимальный пример приложения с PyQt5 + matplotlib, то не могли бы создать тему обсуждения в этом разделе форума и прикрепить архив того примера.

Там по сути нужно правильно продёрнуть виджет в QML, либо перенаследовать некоторые классы, но обычно это делается через наследование от QQuickPaintedItem . Продолжим тогда обсуждение в ветке на форуме.
k
  • 15 червня 2018 р. 06:59

onClicked: { // Вызываем слот калькулятора, чтобы вычесть числа calculator.sub(firstNumber.text, secondNumber.text) //код после вызова слота }
k
  • 15 червня 2018 р. 07:00

почему не выполняется код после вызова слота?

Evgenii Legotckoi
  • 18 червня 2018 р. 03:12

Я вот сейчас банальность скажу, но у меня всё работало. Так что даже и не знаю, надо на код смотреть, что ещё у вас добавлено или отсутствует из библиотек.


P/S/ Извините, вы сейчас всё подряд пробуете или работаете именно с Python?
k
  • 08 жовтня 2018 р. 07:30

Изменив математические действия столкнулся с проблемой - не выводит десятичные дроби

Evgenii Legotckoi
  • 08 жовтня 2018 р. 07:36

Проблема должна быть в описании слота, там целочисленные аргументы

@pyqtSlot(int, int)

В вашем случае, я думаю, так должно быть

@pyqtSlot(float, float)
k
  • 08 жовтня 2018 р. 07:42

заменил, та же картина, причем в python я вывожу переменную 11.5, а в qml вижу 11


Evgenii Legotckoi
  • 08 жовтня 2018 р. 07:44

Вот это ещё замените

sumResult = pyqtSignal(int, arguments=['sum'])

на вот это

sumResult = pyqtSignal(float, arguments=['sum'])
ogustbiller
  • 08 квітня 2020 р. 09:12

Круто! Немного начинает проясняться что к чему. Спасибо.

ogustbiller
  • 01 листопада 2022 р. 09:08

А можно ли из QML сделать привязку свойства к свойству пайтоновского объекта? Ну, т.е. , например, у нашего объекта Calculator обхвялем свойства sumresult и subresult c с декоратором @pyqtProperty, а в QMLе делаем типа такого:

        // Здесь увидим результат сложения
        Text {
            id: sumResult
            text: calculator.sumResult
        }
        ...
                // Здесь увидим результат сложения
        Text {
            id: subResult
            text: calculator.subResult
        }

Коментарі

Only authorized users can post comments.
Please, Log in or Sign up
d
  • dsfs
  • 26 квітня 2024 р. 04:56

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:80бали,
  • Рейтинг балів4
d
  • dsfs
  • 26 квітня 2024 р. 04:45

C++ - Тест 002. Константы

  • Результат:50бали,
  • Рейтинг балів-4
d
  • dsfs
  • 26 квітня 2024 р. 04:35

C++ - Тест 001. Первая программа и типы данных

  • Результат:73бали,
  • Рейтинг балів1
Останні коментарі
k
kmssr08 лютого 2024 р. 18:43
Qt Linux - Урок 001. Автозапуск програми Qt під Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
АК
Анатолий Кононенко05 лютого 2024 р. 01:50
Qt WinAPI - Урок 007. Робота з ICMP Ping в Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
EVA
EVA25 грудня 2023 р. 10:30
Boost - статичне зв&#39;язування в проекті CMake під Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
J
JonnyJo25 грудня 2023 р. 08:38
Boost - статичне зв&#39;язування в проекті CMake під Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
G
Gvozdik18 грудня 2023 р. 21:01
Qt/C++ - Урок 056. Підключення бібліотеки Boost в Qt для компіляторів MinGW і MSVC Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
Тепер обговоріть на форумі
G
George1304 травня 2024 р. 16:13
добавить qlineseries в функции Всем доброго времени суток! Товарищи, помогите, юному падавану обуздать QChart, уже неделю пытаюсь сам решить проблему, в интернете подходящих статей не нашел:) Проблема в следующем:…
PS
Peter Son03 травня 2024 р. 17:57
Best Indian Food Restaurant In Cincinnati OH Ready to embark on a gastronomic journey like no other? Join us at App india restaurant and discover why we're renowned as the Best Indian Food Restaurant In Cincinnati OH . Whether y…
Evgenii Legotckoi
Evgenii Legotckoi02 травня 2024 р. 14:07
Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Добрый день. По моему мнению - да, но то, что будет касаться вызовов к функционалу Андроида, может создать огромные трудности.
IscanderChe
IscanderChe30 квітня 2024 р. 04:22
Во Flask рендер шаблона не передаётся в браузер Доброе утро! Имеется вот такой шаблон: <!doctype html><html> <head> <title>{{ title }}</title> <link rel="stylesheet" href="{{ url_…
G
Gar22 квітня 2024 р. 05:46
Clipboard Как скопировать окно целиком в clipb?

Слідкуйте за нами в соціальних мережах