ДК
Джон Кофи4 марта 2020 г. 19: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
  • 4 марта 2020 г. 19: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);
    ДК
    • 4 марта 2020 г. 20:05

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

      Evgenii Legotckoi
      • 4 марта 2020 г. 20:10

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

        ДК
        • 5 марта 2020 г. 0: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";
        Может нельзя передавать в метод сигнал, а потом соединяться с ним?

          ДК
          • 5 марта 2020 г. 1: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
            • 5 марта 2020 г. 1: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

              Комментарии

              Только авторизованные пользователи могут публиковать комментарии.
              Пожалуйста, авторизуйтесь или зарегистрируйтесь
              Ua

              Qt - Тест 001. Сигналы и слоты

              • Результат:84баллов,
              • Очки рейтинга4
              Ua

              Qt - Тест 001. Сигналы и слоты

              • Результат:42баллов,
              • Очки рейтинга-8
              ОК

              Qt - Тест 001. Сигналы и слоты

              • Результат:47баллов,
              • Очки рейтинга-6
              Последние комментарии
              ИМ
              Игорь Максимов22 ноября 2024 г. 21:51
              Django - Урок 017. Кастомизированная страница авторизации на Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
              Evgenii Legotckoi
              Evgenii Legotckoi31 октября 2024 г. 23:37
              Django - Урок 064. Как написать расширение для Python Markdown Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
              A
              ALO1ZE19 октября 2024 г. 17:19
              Читалка fb3-файлов на Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
              ИМ
              Игорь Максимов5 октября 2024 г. 16:51
              Django - Урок 064. Как написать расширение для Python Markdown Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
              d
              dblas55 июля 2024 г. 20:02
              QML - Урок 016. База данных SQLite и работа с ней в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
              Сейчас обсуждают на форуме
              f
              firstlunoxod15 февраля 2025 г. 13:46
              Рисование на QGraphicsScene при зажатой кнопке мыши Подскажите, пожалуйста! Как данный класс можно дополнить, чтобы созданные объекты можно было перемещать мышкой по сцене?
              Дмитрий
              Дмитрий3 февраля 2025 г. 16:24
              Создание deb-пакета. Как создать ярлык на рабочем столе после установки собственного deb-пакета? Всем привет. Сделал свой deb-пакет с программой. Всё устанавливается и работает. Ставлю по пути /usr/bin/my_application. Как для пользователя при установке пакета сразу создать ярлык на раб…
              NW
              Nayo Wai30 января 2025 г. 19:22
              не запускается компьютер!!! Не запускается компьютер (точнее работает блок , но сам монитор вообще жесть)В общем я ничего с интернета не скачивала в последнее время. На компе никаких левых пр…
              n
              nkly3 января 2025 г. 12:52
              Нужно запретить перемещение только некоторых итемов, остальные перемещать можно. Вопрос решен. Узнать QModelIndex элемента на который мы перетаскиваем другой элемент, можно с помощью функции indexAt(event->position().toPoint()) представления QTreeViev вызываемой в переопр…
              M
              Marsel17 августа 2023 г. 0:26
              OAuth2.0 через VK, получение email Спасибо большое за помощь и простите за то что отнял время своей невнимательностью.

              Следите за нами в социальных сетях