ДК
Джон Кофи04 березня 2020 р. 08:14

Передача сигнала в функцию

Qt, signals

Привет. Можно ли передать в метод сигнал, как параметр и потом сделать коннект на него? код ниже конечно не работает.

_model->waitingFor(SIGNAL(Model::somesignal())); // no matching function for call to..

...
class Model : public QObject
{
    Q_OBJECT
public:
    explicit Model(QObject *parent = nullptr);

    void foo();
    void waitingFor(const char *signal); - не работает
//    void waitingFor(std::function<void()> &signal); - тоже не работает

signals:
    void somesignal();
};
...
void Model::waitingFor(const char *signal)
{
    QEventLoop evloop;
    connect(this, signal, &evloop, &QEventLoop::quit); // не соберется
    connect(this, SIGNAL(signal()), &evloop, SLOT(quit())); // соберется, но QObject::connect: No such signal Model::signal()
    evloop.exec();
}

*так тоже не работает

void Test::waitingFor2(std::function<QMetaMethod> &signal)
{
    QEventLoop evloop;
    connect(this, signal, &evloop, &QEventLoop::quit);
    evloop.exec();
}
Рекомендуємо хостинг TIMEWEB
Рекомендуємо хостинг TIMEWEB
Стабільний хостинг, на якому розміщується соціальна мережа EVILEG. Для проектів на Django радимо VDS хостинг.

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

6
Evgenii Legotckoi
  • 04 березня 2020 р. 08:51
  • (відредаговано)
  • Відповідь була позначена як рішення.

Добрый день.

Можно передать сигнал, но лучше в виде указателей это всё делать, МАКРОСЫ следует изживать из проекта.

Также здесь следует использовать шаблоны. Конкретно ваш случай можно разрулить так

#ifndef MODEL_H
#define MODEL_H

#include <QEventLoop>

class Model : public QObject
{
    Q_OBJECT
public:
    explicit Model(QObject *parent = nullptr) : QObject(parent) {}

    template<typename TSignal>
    void waitingFor(TSignal&& signal)
    {
        QEventLoop evloop;
        connect(this, signal, &evloop, &QEventLoop::quit);
        evloop.exec();
    }

signals:
    void someSignal();
};

#endif // MODEL_H

И вызов метода

model->waitingFor(&Model::someSignal);
    ДК
    • 04 березня 2020 р. 09:05

    великолепно. я уж начал сомневаться, что делаю решаемую задачу.

      Evgenii Legotckoi
      • 04 березня 2020 р. 09:10

      На макросах тоже можно, но лучше не надо. Кода больше получается, сам код в случае макросов требует применения SFINAE - в общем красивым код не назовёшь. Да и макросы менее надёжны.

        ДК
        • 04 березня 2020 р. 13:54
        • (відредаговано)

        на работе просто собрал, а протестил вот сейчас дома- не отрабатывает коннект.

        class Test : public QObject
        {
            Q_OBJECT
        public:
            explicit Test(QObject *parent = nullptr) : QObject(parent) {}
        
            void foo()
            {
                qDebug() << "foo";
                emit somesignal();
            }
        
            template<typename TSignal>
            void waitingFor(TSignal &&signal)
            {
                qDebug() << "waitingFor";
                connect(this, signal, [](){ qDebug() << "signal received"; });
            }
        
        signals:
            void somesignal();
        };
        ...
        _test->foo();
        _test->waitingFor(&Test::somesignal);
        

        именно qDebug() << "signal received";
        Может нельзя передавать в метод сигнал, а потом соединяться с ним?

          ДК
          • 04 березня 2020 р. 14:20

          я вот думаю, что всё же правильней передавать const char *. Но всё равно тишина...

          class Test : public QObject
          {
              Q_OBJECT
          public:
              explicit Test(QObject *parent = nullptr) : QObject(parent) {}
          
              void foo()
              {
                  QTimer *timer = new QTimer;
                  timer->setSingleShot(true);
                  timer->start(5000);
                  connect(timer, &QTimer::timeout, timer, &QTimer::stop);
                  emit somesignal();
              }
          
              static void waitingFor(QObject *sender, const char *signalFromObj)
              {
                  qDebug() << "waitingFor";
                  QEventLoop loop;
                  connect(sender, signalFromObj, &loop, SLOT(quit()));
                  loop.exec();
              }
          
          signals:
              void somesignal();
          };
          ...
          _test->foo();
          Test::waitingFor(_test, SIGNAL(somesignal()));
          qDebug() << "mainwindow";
          

          qDebug() << "mainwindow"; не отработает.

            Evgenii Legotckoi
            • 04 березня 2020 р. 14:38
            • (відредаговано)

            я вот думаю, что всё же правильней передавать const char *. Но всё равно тишина...

            Вы неправильно думаете. При компиляции проектов на C/C++ макросы разворачиваются в полноценный код, а в случае Qt присутствует ещё и этап создания moc файлов, в которых эти макросы уже интерпретируются в сигнатуры функций и создаётся код коннектов. И только потом уже происходит компиляция всех созданных исходников. В Qt не создаются налёту подключения сигналов и слотов из указателей на const char *, потому что SIGNAL и SLOT - это не строки, это макросы. Весь процесс создания кода в случае SIGNAL и SLOT для сигналов и слотов происходит на этапе ПРЕДКОМПИЛЯЦИИ, а не в рантайме. А то, что вы пытаетесь передавать указатель на const char - это рантайм.

            Я вам дал полностью рабочий пример кода, который оттестировал и запустил.

            Вот модифицированный вариант, который тоже работает.

            #ifndef MODEL_H
            #define MODEL_H
            
            #include <QEventLoop>
            #include <QDebug>
            
            class Model : public QObject
            {
                Q_OBJECT
            public:
                explicit Model(QObject *parent = nullptr) : QObject(parent) {}
            
                template<typename TSignal>
                void waitingFor(TSignal&& signal)
                {
                    connect(this, signal, [](){ qDebug() << "Signal emitted"; });
                }
            
            signals:
                void someSignal();
            };
            
            #endif // MODEL_H
            
            

            Вот так его можно использовать

            widget.h

            #ifndef WIDGET_H
            #define WIDGET_H
            
            #include <QWidget>
            #include "model.h"
            
            QT_BEGIN_NAMESPACE
            namespace Ui { class Widget; }
            QT_END_NAMESPACE
            
            class Widget : public QWidget
            {
                Q_OBJECT
            
            public:
                Widget(QWidget *parent = nullptr);
                ~Widget();
            
            private slots:
                void on_pushButton_clicked();
            
                void on_sendSignalButton_clicked();
            
            private:
                Ui::Widget *ui;
            
                Model m_model;
            };
            #endif // WIDGET_H
            
            

            widget.cpp

            #include "widget.h"
            #include "ui_widget.h"
            
            Widget::Widget(QWidget *parent)
                : QWidget(parent)
                , ui(new Ui::Widget)
            {
                ui->setupUi(this);
            }
            
            Widget::~Widget()
            {
                delete ui;
            }
            
            void Widget::on_pushButton_clicked()
            {
                m_model.waitingFor(&Model::someSignal);
            }
            
            void Widget::on_sendSignalButton_clicked()
            {
                emit m_model.someSignal();
            }
            
            

            Вот тестовый проект

            CheckSignals.zip CheckSignals.zip

              Коментарі

              Only authorized users can post comments.
              Please, Log in or Sign up
              AD

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

              • Результат:50бали,
              • Рейтинг балів-4
              m
              • molni99
              • 26 жовтня 2024 р. 01:37

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

              • Результат:80бали,
              • Рейтинг балів4
              m
              • molni99
              • 26 жовтня 2024 р. 01:29

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

              • Результат:20бали,
              • Рейтинг балів-10
              Останні коментарі
              ИМ
              Игорь Максимов22 листопада 2024 р. 11:51
              Django - Підручник 017. Налаштуйте сторінку входу до Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
              Evgenii Legotckoi
              Evgenii Legotckoi31 жовтня 2024 р. 14:37
              Django - Урок 064. Як написати розширення для Python Markdown Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
              A
              ALO1ZE19 жовтня 2024 р. 08:19
              Читалка файлів fb3 на Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
              ИМ
              Игорь Максимов05 жовтня 2024 р. 07:51
              Django - Урок 064. Як написати розширення для Python Markdown Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
              d
              dblas505 липня 2024 р. 11:02
              QML - Урок 016. База даних SQLite та робота з нею в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
              Тепер обговоріть на форумі
              Evgenii Legotckoi
              Evgenii Legotckoi24 червня 2024 р. 15:11
              добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
              t
              tonypeachey115 листопада 2024 р. 06:04
              google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
              NSProject
              NSProject04 червня 2022 р. 03:49
              Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…
              9
              9Anonim25 жовтня 2024 р. 09:10
              Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…

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