Arrow
Arrow12 января 2017 г. 13:53

Отслеживание нажатия клавиш в QTableView

keyPressEvent, QTableView

Подскажите как можно в QTableView реализовать отслеживание нажатия клавиш.

Версия Qt 5.7.

Для решения задачи решил наследоваться от QTableView и переопределить метод keyPressEvent. ниже привожу свой код и подробное описание, что делал.

Файл mtableview.h:

#ifndef MTABLEVIEW_H
#define MTABLEVIEW_H

#include <QObject>
#include <QTableView>
#include <QKeyEvent>

class QTableViewPrivate;

class MTableView : public QTableView
{
public:
    MTableView(QWidget *parent = 0);
    MTableView(QTableViewPrivate &d, QWidget *parent);
signals:
    void keyPressEvent(QKeyEvent *event);
};

#endif // MTABLEVIEW_H

Файл mtableview.cpp:

#include "mtableview.h"

MTableView::MTableView(QWidget *parent):
    QTableView(parent)
{ }

MTableView::MTableView(QTableViewPrivate &d, QWidget *parent)
    : QTableView(d, parent)
{ }

void MTableView::keyPressEvent(QKeyEvent *event)
{
    QTableView::keyPressEvent(event);
}

После чего в дизайнере для элемента QTableView на форме указал "Преобразовать в" и указал свой класс MTableView (все работает)

Ошибки начались при реализации метода keyPressEvent в программе:

connect(ui->lView, &MTableView::keyPressEvent, this, &Wnd::keyPressEvent);
void Wnd::keyPressEvent(QKeyEvent *event)
{
    if(event->key() == Qt::Key_Return)
    {
        on_lView_doubleClicked(ui->lView->currentIndex());
    }
}

Выдает ошибки:

/gcc_64/include/QtCore/qobject.h:-1: In instantiation of 'static QMetaObject::Connection QObject::connect(const typename QtPrivate::FunctionPointer<Func>::Object*, Func1, const typename QtPrivate::FunctionPointer<Func2>::Object*, Func2, Qt::ConnectionType) [with Func1 = void (MTableView::*)(QKeyEvent*); Func2 = void (FManagerWnd::*)(QKeyEvent*); typename QtPrivate::FunctionPointer<Func>::Object = MTableView; typename QtPrivate::FunctionPointer<Func2>::Object = FManagerWnd]':

/gcc_64/include/QtCore/qglobal.h:746: ошибка: static assertion failed: No Q_OBJECT in the class with the signal
 #define Q_STATIC_ASSERT_X(Condition, Message) static_assert(bool(Condition), Message)
                                               ^

/gcc_64/include/QtCore/qobject.h:226: in expansion of macro 'Q_STATIC_ASSERT_X'
         Q_STATIC_ASSERT_X(QtPrivate::HasQ_OBJECT_Macro<typename SignalType::Object>::Value,
         ^

Не могу разобраться, что не так.

P.S. Извините за столько текста :)

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

Вам это нравится? Поделитесь в социальных сетях!

6
Evgenii Legotckoi
  • 12 января 2017 г. 13:59

А вот тут Вы сударь пролетели неплохо.

keyPressEvent - это метод события. А Вы его как сигнал определяете в заголовочном файле mtableview.h . Да ещё и подключаете его опять же к методу keyPressEvent

В общем нельзя так, keyPressEvent - это не сигнал и не слот, к нему коннект никаким боком не прикрутишь. Нужно объявлять свои сигналы и слоты для передачи инфы между объектами. А в этом методе можно только ловить события, которые прилетают из приложения через event`ы

    Arrow
    • 12 января 2017 г. 14:16

    Хорошо, только я не пойму как именно это реализовать. Основное это не могу понять как свой сигнал заставить срабатывать при нажатии клавиши.

      Evgenii Legotckoi
      • 12 января 2017 г. 23:17
      • Ответ был помечен как решение.

      Из того, что я вижу в коде, можно сделать следующие выводы о том, как это дело можно реализовать.

      А именно, объявить сигнал, который будет испускаться при нажатий клавиш в таблице. И объявить слот в классе Wnd , который уже будет принимать этот сигнал.

      Файл mtableview.h:
      #ifndef MTABLEVIEW_H
      #define MTABLEVIEW_H
      
      #include <QObject>
      #include <QTableView>
      #include <QKeyEvent>
      
      class QTableViewPrivate;
      
      class MTableView : public QTableView
      {
      public:
          MTableView(QWidget *parent = 0);
          MTableView(QTableViewPrivate &d, QWidget *parent);
      signals:
          void keyReturnPressed();
      protected:
          virtual void keyPressEvent(QKeyEvent *event) override;
      };
      
      #endif // MTABLEVIEW_H
      
      Файл mtableview.cpp:
      #include "mtableview.h"
      
      MTableView::MTableView(QWidget *parent):
          QTableView(parent)
      { }
      
      MTableView::MTableView(QTableViewPrivate &d, QWidget *parent)
          : QTableView(d, parent)
      { }
      
      void MTableView::keyPressEvent(QKeyEvent *event)
      {
          if(event->key() == Qt::Key_Return)
          {
              emit keyReturnPressed();
          }
          QTableView::keyPressEvent(event);
      }
      

      В заголовочном файле Wnd необходимо объявить слот, который уже примет этот сигнал от таблицы.

      public slots:
          void slotKeyReturnPressed();
      

      В файле cpp написать реализацию этого слота.

      void Wnd::slotKeyReturnPressed()
      {
          // ToDo something
      }
      

      И подключить сигнал к слоту

      connect(ui->lView, &MTableView::keyReturnPressed, this, &Wnd::slotKeyReturnPressed);
      

      Ну и почитайте на досуге вот эту статью про сигналы и слоты в Qt5

        Arrow
        • 13 января 2017 г. 15:13

        Спасибо. Статья хорошая, я ее как-то пропустил :) .

        Решение немного подкорректировал так:

        В классе наследнике:

        void MTableView::keyPressEvent(QKeyEvent *event)
        {
            if(event->key() == Qt::Key_Return || event->key() == Qt::Key_Backspace || event->key() == Qt::Key_Tab)
            {
                emit keyPressed(event);
            }
            QTableView::keyPressEvent(event);
        }
        

        И далее:

        connect(ui->lTableView, &MTableView::keyPressed, this, &Wnd::keyPress);
        
        void Wnd::keyPress(QKeyEvent *event)
        {
            if(event->key() == Qt::Key_Return)
            {
                on_lTableView_doubleClicked(ui->lTableView->currentIndex());
            }
            else if(event->key() == Qt::Key_Backspace)
            {
                on_lPushButton_clicked();
            }
            else
            {
                ui->lBit->setFocus();
            }
        }
        

        Спасибо за помощь!

          Evgenii Legotckoi
          • 14 января 2017 г. 1:24

          Не нравится мне ваш вариант кода ))) Давайте сразу приведём его в более грамотный вид:

          void MTableView::keyPressEvent(QKeyEvent *event)
          {
              switch (event->key()) 
              {
              case Qt::Key_Return:
              case Qt::Key_Backspace:
              case Qt::Key_Tab:
                  emit keyPressed(event);
                  break;
              default:
                  break;
              }
              QTableView::keyPressEvent(event);
          }
          

          И далее:

          connect(ui->lTableView, &MTableView::keyPressed, this, &Wnd::keyPress);
          
          void Wnd::keyPress(QKeyEvent *event)
          {
              switch (event->key()):
              {
              case Qt::Key_Return:
                  on_lTableView_doubleClicked(ui->lTableView->currentIndex());
                  break;
              case Qt::Key_Backspace:
                  on_lPushButton_clicked();
                  break;
              default:
                  ui->lBit->setFocus();
                  break;
              }
          }
          

          А теперь, чем мой вариант лучше:

          1. В данном случае метод key() вызывается всего один раз, в вашем коде он вызывается каждый раз при каждом условии.
          2. Конструкция switch case как правило разворачивается компилятором в более эффективный ассемблерный код. К сожалению я не могу привести статью в подтверждение своих слов, но на хабре такая статья есть, просто я её уже забыл. Где разбирается этот момент.
          3. В условии проверяются enum константы, которые традиционно проверяются через switch case
          4. Сама по себе конструкция switch case является более прозрачной для понимания, чем if else
            Arrow
            • 14 января 2017 г. 5:46

            Спасибо большое!

            Статьи такой не встречал, попробую найти. /p>

            Согласен ваш вариант более читабельнее и понятнее, к тому же выглядит элегантнее.

              Комментарии

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

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

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

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

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

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

              • Результат:20баллов,
              • Очки рейтинга-10
              Последние комментарии
              i
              innorwall11 ноября 2024 г. 22:12
              Django - Урок 055. Как написать функционал auto populate field Freckles because of several brand names retin a, atralin buy generic priligy
              i
              innorwall11 ноября 2024 г. 18:23
              QML - Урок 035. Использование перечислений в QML без C++ priligy cvs 24 Together with antibiotics such as amphotericin B 10, griseofulvin 11 and streptomycin 12, chloramphenicol 9 is in the World Health Organisation s List of Essential Medici…
              i
              innorwall11 ноября 2024 г. 15:50
              Qt/C++ - Урок 052. Кастомизация Qt Аудио плеера в стиле AIMP It decreases stress, supports hormone balance, and regulates and increases blood flow to the reproductive organs buy priligy online safe Promising data were reported in a PDX model re…
              i
              innorwall11 ноября 2024 г. 14:19
              Алгоритм сортировки кучей The role of raloxifene in preventing breast cancer priligy precio
              i
              innorwall11 ноября 2024 г. 13:55
              PyQt5 - Урок 006. Работа с QTableWidget buy priligy 60 mg 53 have been reported by Javanovic Santa et al
              Сейчас обсуждают на форуме
              i
              innorwall11 ноября 2024 г. 20:56
              добавить qlineseries в функции buy priligy senior brother Chu He, whom he had known for many years
              i
              innorwall11 ноября 2024 г. 10:55
              Всё ещё разбираюсь с кешем. priligy walgreens levitra dulcolax carbs The third ring was found to be made up of ultra relativistic electrons, which are also present in both the outer and inner rings
              9
              9Anonim25 октября 2024 г. 9:10
              Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…
              ИМ
              Игорь Максимов3 октября 2024 г. 4:05
              Реализация навигации по разделам Спасибо Евгений!

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