‘виртуальный логический QAbstractScrollArea::eventFilter(QObject*, QEvent*)’ защищен

Я пытался найти ответ на этот вопрос с помощью Google, но ничего не дало. Я нахожусь в процессе преобразования приложения с Qt4 на Qt5. Это приложение отлично скомпилировано в Qt4, но когда я пытаюсь скомпилировать Qt5, оно дает мне эту ошибку разрешения. Поскольку статус этого класса защищен в обеих версиях, мне трудно понять, что мне нужно изменить.

Эта проблема компиляции была воспроизведена в нескольких разных установках Ubuntu (включая wsl), но я еще не пробовал в Fedora.

Вот подмножество класса

#include <QWidget>
#include <QEvent>
#include <QTableWidget>
#include <QItemDelegate>
#include <QModelIndex>
#include <QSize>
#include <qdialog.h>
#include <qcombobox.h>
#include "ui_pegs_page.h"
#include <string>

class EGS_ConfigReader;
class QProcess;
class PEGS_RunOutput;
class QTableWidget;

struct Element {
  int   Z;
  std::string symbol;
 float  aw;
 float  Iev;
 float  rho;
};

const int n_element = 100;

extern Element element_data[];

class TableEventHandler : public QObject {
    Q_OBJECT
public:
    TableEventHandler(QTableWidget *parent);
protected:
    bool eventFilter(QObject *o, QEvent *e);
private:
    QStringList itemCopy;
    QList<QTableWidgetSelectionRange> copyRange;
};

Редактировать:

Вот метод, который имеет проблемы.

TableEventHandler::TableEventHandler(QTableWidget *parent) :
  QObject(parent) {
  if( parent == 0 )
   qFatal("TableEventHandler::TableEventHandler: parent can not be null!");
}


bool TableEventHandler::eventFilter(QObject *o, QEvent *e) {
  if( !o ) qWarning("TableEventHandler::eventFilter called with 0 object?");
  if( QString(o->metaObject()->className()) != tr("QTableWidget") ) {
#ifdef EI_DEBUG
      qDebug("Only QTableWidget objects accepted! Returning!");
#endif
      return false;
  }
  QTableWidget *to = (QTableWidget *)o;
  if( e->type() == QEvent::KeyPress ) {
    QKeyEvent *ke = (QKeyEvent*)e;
    if(ke->matches(QKeySequence::Copy) ){
       QString cellText; itemCopy.clear(); copyRange.clear();
       QList<QTableWidgetSelectionRange> ts = to->selectedRanges();
       if(!ts.isEmpty()) {
          for ( int irow = ts.first().topRow(); irow <= ts.first().bottomRow(); irow++){
               for ( int icol = ts.first().leftColumn(); icol <= ts.first().rightColumn(); icol++){
                   QTableWidgetItem *w = to->item(irow,icol);
                   if(w) cellText = w->text();
                   if ( !cellText.isEmpty() ){
                      itemCopy << cellText;
                   }
                   else
                      itemCopy << " ";
               }
          }
          copyRange = ts;
          //cout << itemCopy.join(", ").toLatin1().data() << endl;
       }
       else {
            QTableWidgetItem *w = to->item(to->currentRow(), to->currentColumn());
            if (w) cellText = w->text();
            if ( !cellText.isEmpty() )
                 itemCopy << cellText;
            else itemCopy << "";
       }
       return true;
    }
    else if(ke->matches(QKeySequence::Paste) && !itemCopy.isEmpty() && !copyRange.isEmpty()){
       QList<QTableWidgetSelectionRange> cs = to->selectedRanges();
       int top = cs.first().topRow(), left = cs.first().leftColumn(), icount = 0;
       QTableWidgetSelectionRange ts = QTableWidgetSelectionRange(
                                       top , left,
                                       top  + copyRange.first().rowCount()-1,
                                       left + copyRange.first().columnCount()-1);
       for ( int irow = ts.topRow(); irow <= ts.bottomRow(); irow++){
         for ( int icol = ts.leftColumn(); icol <= ts.rightColumn(); icol++){
             if ( ++icount <= itemCopy.size() )
                to->setItem(irow, icol, new QTableWidgetItem(itemCopy[icount-1]));
                to->setItem(irow, icol, new QTableWidgetItem(itemCopy[icount-1]));
         }
       }
       return true;
    }
    else if(ke->matches(QKeySequence::Cut) ){
       QString cellText; itemCopy.clear(); copyRange.clear();
       QList<QTableWidgetSelectionRange> ts = to->selectedRanges();
       if(!ts.isEmpty()) {
         for (int irow = ts.first().topRow(); irow <= ts.first().bottomRow(); irow++) {
           for(int icol = ts.first().leftColumn(); icol <= ts.first().rightColumn(); icol++) {
               QTableWidgetItem *w = to->item(irow,icol);
               if(w) cellText = w->text();
               if ( !cellText.isEmpty() ){
                  itemCopy << cellText;
               }
               else
                  itemCopy << "";
               to->setItem(irow,icol,0);
           }
         }
         copyRange = ts;
         //cout << itemCopy.join(", ").toLatin1().data() << endl;
       }
       return true;
    }
    else if(ke->matches(QKeySequence::Delete) ){
       QList<QTableWidgetSelectionRange> ts = to->selectedRanges();
       if(!ts.isEmpty()) {
         for (int irow = ts.first().topRow(); irow <= ts.first().bottomRow(); irow++) {
           for(int icol = ts.first().leftColumn(); icol <= ts.first().rightColumn(); icol++) {
               to->setItem(irow,icol,0);
           }
         }
       }
       return true;
    }
    else
        to->eventFilter(o, e);

  }
  return false;
}

person crcrewso    schedule 14.02.2017    source источник


Ответы (1)


Вы также обращаетесь к защищенному методу QAbstractScrollArea::eventFilter(QObject*, QEvent*)

  1. из метода класса, который не наследуется от QAbstractScrollArea (скорее всего), или
  2. из метода какого-то другого класса, который не является другом QAbstractScrollArea (маловероятно), или
  3. из какой-то функции, которая не является другом QAbstractScrollArea (маловероятно).

Обратите внимание, что TableEventHandler наследуется непосредственно от QObject, а не от QAbstractScrollArea. Итак, если вы пытаетесь вызвать QAbstractScrollArea::eventFilter(QObject*, QEvent*) из одного из методов TableEventHandler, вы получите эту ошибку.

EDIT: Глядя на ваш отредактированный ответ, я вижу, что вы звоните

to->eventFilter(o, e);

в TableEventHandler::eventFilter(QObject *o, QEvent *e), где QTableWidget *to = (QTableWidget *)o;. Программист, вероятно, имел в виду, что TableEventHandler::eventFilter не фильтрует соответствующее событие в этот момент. Вместо этого метод должен просто вернуть false, чтобы передать управление любым другим фильтрам событий, установленным позже для этого объекта.

person jotik    schedule 14.02.2017
comment
Я вижу, к чему вы клоните, и буду исследовать. Мне интересно, как, если это проблема наследования, это не возникло в qt4? - person crcrewso; 15.02.2017