"QComboBox弹出窗口"扩展和QtWebkit

9

在Firefox/Chrome/InternetExplorer/Safari/Opera中,来自下拉框的弹出窗口会随着内容而扩展,如Firefox图片所示:

Firefox combobox

QComboBox的弹出窗口不会扩展内容。由于QComboBox大小的限制,弹出窗口受到限制,见QWebView图片:

Qt and QtWebkit combobox

因此我实现了QComboBox::showPopup

void newQComboBox::showPopup() {
    int width = this->width();
    this->view()->setTextElideMode( Qt::ElideNone );

    const int iconSize = this->iconSize().width();
    const QFontMetrics fontMetrics = this->fontMetrics();
    const int j = this->count();

    for( int i=0; i < j; ++i ) {
        const int textWidth = fontMetrics.width( this->itemText(i) + "WWW" );
        if (this->itemIcon(i).isNull()) {
            width = qMax(width, textWidth);
        } else {
            width = qMax(width, textWidth + iconSize);
        }
    }

    QStyleOptionComboBox opt;
    this->initStyleOption(&opt);
    QSize size(width, 0);
    size = this->style()->sizeFromContents(QStyle::CT_ComboBox, &opt, size, this);

    this->view()->setFixedWidth( width );

    QComboBox::showPopup();
}

有没有办法修改(重新实现)QtWebkit中QWebViewsQComboBox :: showPopup Qt-BUG(建议):https://bugreports.qt.io/browse/QTBUG-35771

你在使用什么样式?弹出框的显示方式以及匹配视图是否具有与组合框相同的尺寸或允许更大,完全取决于使用的样式。 - peppe
2
这也告诉你如何修复它:查看QComboBox::showPopup代码并使用一个QProxyStyle来转移正确的调用。当然,所有这些都是假设QWebView正在使用QComboBox,而不是类似于QComboBox的其他东西... - peppe
你是否可以访问你期望显示这个“ComboBox”的HTML文件,或者你无法控制“QWebView”所显示的内容?因为我实现了一个使用你自定义的“QComboBox”的ComboBox到“QWebView”,但它只能在HTML页面使用特定代码而不是“<select>”代码来显示“ComboBox”。 - Antonio Dias
@user2014561 我正在开发一个浏览器而不是特定的应用程序。我需要访问任何网站,就像Firefox/InternetExplorer/Chrome浏览器一样(我的意思是下拉框,就像我发布的截图:http://i.stack.imgur.com/VOn9w.png)。谢谢。 - Guilherme Nascimento
@peppe感谢您的评论,我找到了解决方案。 - Guilherme Nascimento
2个回答

6

我使用QProxyStyle类解决了这个问题,以下是示例:

#include <QProxyStyle>
#include <QAbstractItemView>
#include <QComboBox>

class myProxyStyle : public QProxyStyle
{

public:
    int styleHint(StyleHint hint, const QStyleOption *option, const QWidget *widget, QStyleHintReturn *returnData) const
    {
        if(hint == QStyle::SH_ComboBox_Popup) {
            const QComboBox *combo = (QComboBox *) widget;//cast to QComboBox
            const QObjectList a = combo->children();
            const int j = a.count();
            QAbstractItemView *view = 0;
            QString className = "";
            bool hasView = false;

            /*
            at this point I have not used combo->view() because he "crash" the application without explanation
            so I had to make a loop to find the "view"
            */
            for (int i = 0; i < j; ++i) {
                const QObjectList b = a.at(i)->children();
                const int y = b.count();

                for (int x = 0; x < y; ++x) {
                    className = b.at(x)->metaObject()->className();

                    if (className == "QComboBoxListView") {
                        view = (QAbstractItemView *) b.at(x);
                        hasView = true;
                        break;
                    }
                }

                if (hasView) {
                    break;
                }
            }

            if (hasView) {
                const int iconSize = combo->iconSize().width();
                const QFontMetrics fontMetrics1 = view->fontMetrics();
                const QFontMetrics fontMetrics2 = combo->fontMetrics();
                const int j = combo->count();
                int width = combo->width(); //default width

                for (int i = 0; i < j; ++i) {
                    const int textWidth = qMax(
                        fontMetrics1.width(combo->itemText(i) + "WW"),
                        fontMetrics2.width(combo->itemText(i) + "WW")
                    );

                    if(combo->itemIcon(i).isNull()) {
                        width = qMax(width, textWidth);
                    } else {
                        width = qMax(width, textWidth + iconSize);
                    }
                }

                view->setFixedWidth(width);
            }
        }

        return QProxyStyle::styleHint(hint, option, widget, returnData);
    }
};

使用方法:

您必须在 QApplication 中应用(通常是文件 main.cpp):

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    a.setStyle(new myProxyStyle);//set ProxyStyle

    return a.exec();
}

@peppe 感谢您的评论,我找到了解决方案。


-2
请尝试使用setMinimumWidth函数,并传递您计算出的最大宽度。它应该能够正常工作。 ui.comboBox->view()->setMinimumWidth(width); 有关更多详细信息,请查看https://qt-project.org/forums/viewthread/26388链接。

1
组合框由 Web 视图显示。作者已经有一个可行的实现,问题是如何让 Web 视图使用他的实现。 - Kuba hasn't forgotten Monica
1
我同意@Kuba的观点。 - László Papp

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接