7

ここやqtcentreのような他の場所を検索すると、この問題が発生したことがわかりましたが、機能していないようです。2 つのPaneウィジェット ( QFrame からサブクラス)を含むQSplitterを備えた MainWindow ウィジェットがあります各ペインには、同一のQActions/Shortcutsが関連付けられたメニューバーがあります。

ShortcutContextssetShortcutContext()のすべての組み合わせを試しました。

WindowShortcutおよびApplicationShortcutコンテキストは、予想される「あいまいなショートカットの過負荷」を与えます。

WidgetShortcutとWidgetWithChildrenShortcutどちらも何もしません。

メニューを手動で有効にすると、もちろん正常に機能します。また、オーバーロードされたenterEvent()を使用して、親ウィジェットにフォーカスを強制しようとしました。

何か案は?

ありがとう。

main.h

#include <QMainWindow>
#include <QFrame>

QT_BEGIN_NAMESPACE
class QAction;
class QMenu;
class QHBoxLayout;
class QSplitter;
class QWidget;
QT_END_NAMESPACE

class Pane: public QFrame
{
  Q_OBJECT

  public:
    Pane(QWidget* parent = 0);

  protected:
    void            enterEvent(QEvent *event);
    void            leaveEvent(QEvent *event);

  private:
    void            createMenus();

    QMenuBar *      m_menuBar;

  private Q_SLOTS:
    void            split();
};

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    MainWindow();

private:
    void createActions();
    void createMenus();
    void setupUi(QMainWindow *MainWindow);

    QMenu *fileMenu;
    QAction *exitAct;

    QWidget *centralwidget;
    QHBoxLayout *horizontalLayout;
    QSplitter *splitter;
    QFrame *frame;
    QFrame *frame_2;
};

main.cpp

#include <iostream>
#include <QApplication>
#include <QMainWindow>
#include <QSplitter>
#include <QFrame>
#include <QMenuBar>
#include <QBoxLayout>
#include "main.h"

Pane::Pane(QWidget* parent) :
    QFrame(parent)
{
    setFrameShape(QFrame::StyledPanel);
    setFrameShadow(QFrame::Raised);

    QVBoxLayout *layout = new QVBoxLayout;
    QFrame::setLayout(layout);

    m_menuBar = new QMenuBar;
    QWidget *m_widget = new QWidget;

    layout->addWidget(m_menuBar);
    layout->addWidget(m_widget);
    layout->setContentsMargins(2, 2, 2, 2);

    show();

    createMenus();
}

void
Pane::enterEvent(QEvent *event)
{   
    std::cout << "enter" << std::endl;
    setFocus();
    setStyleSheet("QFrame { border: 1px solid rgb(127, 127, 0); }");
    if (focusWidget())
        std::cout << "focuswidget = " << focusWidget()->objectName().toUtf8().constData() << std::endl;
}

void
Pane::leaveEvent(QEvent *event)
{   
    std::cout << "leave" << std::endl;
    clearFocus();
    setStyleSheet("QFrame { border: 1px solid rgb(64, 64, 64); }");
}

void
Pane::split()
{
    std::cout << "split pane" << std::endl;
}

void
Pane::createMenus()
{
    QMenu *paneMenu = m_menuBar->addMenu(tr("&Pane"));

    QAction *paneSplitAct = new QAction(tr("Split"), this);
    paneSplitAct->setShortcut(Qt::Key_S);
    paneSplitAct->setShortcutContext(Qt::WidgetWithChildrenShortcut);
    paneSplitAct->setStatusTip(tr("Split Pane"));
    connect(paneSplitAct, SIGNAL(triggered()), this, SLOT(split()));
    paneMenu->addAction(paneSplitAct);
}

MainWindow::MainWindow()
{
    setupUi(this);

    createActions();
    createMenus();
}

void MainWindow::createActions()
{
    exitAct = new QAction(tr("E&xit"), this);
    exitAct->setShortcuts(QKeySequence::Quit);
    exitAct->setStatusTip(tr("Exit the application"));
    connect(exitAct, SIGNAL(triggered()), this, SLOT(close()));
}

void MainWindow::createMenus()
{
    fileMenu = menuBar()->addMenu(tr("&File"));
    fileMenu->addAction(exitAct);
}


void MainWindow::setupUi(QMainWindow *MainWindow)
{
    if (MainWindow->objectName().isEmpty())
        MainWindow->setObjectName(QString::fromUtf8("MainWindow"));

    MainWindow->resize(800, 600);
    centralwidget = new QWidget(MainWindow);
    centralwidget->setObjectName(QString::fromUtf8("centralwidget"));
    horizontalLayout = new QHBoxLayout(centralwidget);
    horizontalLayout->setObjectName(QString::fromUtf8("horizontalLayout"));
    splitter = new QSplitter(centralwidget);
    splitter->setObjectName(QString::fromUtf8("splitter"));
    splitter->setOrientation(Qt::Horizontal);

    frame = new Pane(splitter);
    frame->setObjectName(QString::fromUtf8("frame"));
    splitter->addWidget(frame);

    frame_2 = new Pane(splitter);
    frame_2->setObjectName(QString::fromUtf8("frame_2"));
    splitter->addWidget(frame_2);

    horizontalLayout->addWidget(splitter);

    MainWindow->setCentralWidget(centralwidget);

    QMetaObject::connectSlotsByName(MainWindow);
}

int main(int argc, char *argv[])
{
    QApplication app(argc, argv);
    app.setOrganizationName("Trolltech");
    app.setApplicationName("Application Example");
    MainWindow mainWin;
    mainWin.show();
    return app.exec();
}

main.pro

HEADERS       = main.h
SOURCES       = main.cpp
CONFIG       += no_keywords

更新:コンテキストの使用と組み合わせてaddAction(paneSplitAct)最後に呼び出しを 追加すると、私が望むものが得られるようです。 Pane::createMenus()Qt::WidgetShortcut

ドキュメントについて私が理解していることから、これはウィジェットにコンテキストメニューを作成することになっています。私はそれを取得していないようです(マウスの右クリックだと思います)が、それは必要ないので問題ありません。フォーカスを正しく設定するには、eventEvent()とのleaveEvent()オーバーライドが引き続き必要です。

4

4 に答える 4

9

同様の問題がありました.2つの異なるウィジェットにアクションの同じショートカットがありました(ウィジェット間で異なるアクション)。ウィジェットの 1 つだけがアプリケーションに表示されている限り、すべてが機能しました。両方が表示されるとすぐに、「あいまいなショートカットの過負荷」というメッセージが表示されました。

解決策は次のとおりです。両方のアクションのコンテキストを Qt::WidgetWithChildrenShortcut に正しく設定する必要があります。その後、現在のフォーカスに従って、正常に機能しました。

正しいコンテキストを持つアクションが 1 つだけの場合、デフォルトのコンテキストを持つアクションがトリガーされ、メッセージが表示されました。

両方のアクションにコンテキストが設定されていない場合 (デフォルト)、最後に作成されたアクションがトリガーされ、メッセージが表示されます。

そのため、ショートカットを含むアクションを任意の場所に追加する場合は、正しいコンテキストと設定について考え、制限を設けることを忘れないでください!

于 2014-01-17T16:36:22.253 に答える
1

このようなほとんどのシナリオでは、ショートカットコンテキストをに設定するWidgetShortcutのが正しいことです。ただし問題は、重複したアクションが (従来のウィジェットの意味で) フォーカスできないメニュー バーにあることです。これが、何もしない理由です。

共有アクションをメイン ウィンドウに配置して、アプリケーションのショートカットにする方が理にかなっている場合があります。次に、アクションがトリガーするメイン ウィンドウ スロットで、どのPaneオブジェクトにフォーカスがあるかを判断し、コマンドをそれにプッシュします。

于 2012-04-27T06:48:47.660 に答える
1

このソリューションは、メニューバーが無効になっているか使用されていないときに、ショートカットでアクションをトリガーするのにうまく機能しました。

ショートカットを追加するルーチン:

void StingrayEditor::add_shortcut(const QJsonObject& item_json)
{
    QString item_path = item_json["path"].toString();
    QString shortcut = item_json["shortcut"].toString();
    if (!shortcut.isEmpty()) {
        QKeySequence key_sequence = QKeySequence::fromString(shortcut);
        QAction* shortcut_action = new QAction(item_path, this);
        if (!key_sequence.isEmpty()) {
            shortcut_action->setShortcut(key_sequence);
            shortcut_action->setShortcutContext(Qt::ApplicationShortcut);
        }
        connect(shortcut_action, &QAction::triggered, this, [item_path]()
        {
            // Action to be executed
        });

        // Add the action to the main window.
        addAction(shortcut_action);
    }
}

ご利用いただくことが大切ですshortcut_action->setShortcutContext(Qt::ApplicationShortcut);

次に、イベントをフィルター/リッスンしてキャッチする必要がありますQEvent::Shortcut

bool StingrayEditor::eventFilter(QObject* obj, QEvent* e)
{
    switch (e->type()) {
    case QEvent::Shortcut: {
        QShortcutEvent* sev = static_cast<QShortcutEvent*>(e);
        if (sev->isAmbiguous()) {
            foreach(const auto& action, actions()) {
                if (action->shortcut() == sev->key()) {
                    action->trigger(); // Trigger the action that matches the ambigous shortcut event.
                    return true;
                }
            }
        }
    }
    // ...
    default: break;
    }

    return false;
}

次のようなイベントに登録することを忘れないでください。

qApp->installEventFilter(this);

于 2017-08-09T17:12:23.827 に答える
0

「オーバーロード」は、このショートカットを複数のアクションに使用したことを意味します。各アクション ショートカットは一意である必要があります。

于 2021-06-10T05:30:59.373 に答える