/* * Copyright 2008 Benjamin C. Meyer * Taken from Arora Browser. */ #include "SearchLineEdit.h" #include #include #include #include #include ExLineEdit::ExLineEdit(QWidget *parent) : QWidget(parent) , m_leftWidget(0) , m_lineEdit(new QLineEdit(this)) , m_clearButton(0) { setFocusPolicy(m_lineEdit->focusPolicy()); setAttribute(Qt::WA_InputMethodEnabled); setSizePolicy(m_lineEdit->sizePolicy()); setBackgroundRole(m_lineEdit->backgroundRole()); setMouseTracking(true); setAcceptDrops(true); setAttribute(Qt::WA_MacShowFocusRect, true); QPalette p = m_lineEdit->palette(); setPalette(p); // line edit m_lineEdit->setFrame(false); m_lineEdit->setFocusProxy(this); m_lineEdit->setAttribute(Qt::WA_MacShowFocusRect, false); QPalette clearPalette = m_lineEdit->palette(); clearPalette.setBrush(QPalette::Base, QBrush(Qt::transparent)); m_lineEdit->setPalette(clearPalette); // clearButton m_clearButton = new ClearButton(this); connect(m_clearButton, SIGNAL(clicked()), m_lineEdit, SLOT(clear())); connect(m_lineEdit, SIGNAL(textChanged(const QString&)), m_clearButton, SLOT(textChanged(const QString&))); } void ExLineEdit::setLeftWidget(QWidget *widget) { m_leftWidget = widget; } QWidget *ExLineEdit::leftWidget() const { return m_leftWidget; } void ExLineEdit::resizeEvent(QResizeEvent *event) { Q_ASSERT(m_leftWidget); updateGeometries(); QWidget::resizeEvent(event); } void ExLineEdit::updateGeometries() { QStyleOptionFrameV2 panel; initStyleOption(&panel); QRect rect = style()->subElementRect(QStyle::SE_LineEditContents, &panel, this); int height = rect.height(); int width = rect.width(); int m_leftWidgetHeight = m_leftWidget->height(); int clearButtonWidth = this->height(); switch (layoutDirection()) { case Qt::RightToLeft: m_clearButton->setGeometry(rect.x(), rect.y() + (height - clearButtonWidth) / 2, clearButtonWidth, this->height()); m_lineEdit->setGeometry(m_clearButton->x() + m_clearButton->width(), 0, width - clearButtonWidth - m_leftWidget->width() - 2, this->height()); m_leftWidget->setGeometry(this->width() - m_leftWidget->width() - 2, rect.y() + (height - m_leftWidgetHeight) / 2, m_leftWidget->width(), m_leftWidget->height()); break; case Qt::LeftToRight: default: m_leftWidget->setGeometry(rect.x() + 2,rect.y() + (height - m_leftWidgetHeight) / 2, m_leftWidget->width(), m_leftWidget->height()); m_lineEdit->setGeometry(m_leftWidget->x() + m_leftWidget->width(), 0, width - clearButtonWidth - m_leftWidget->width(), this->height()); m_clearButton->setGeometry(this->width() - clearButtonWidth, 0, clearButtonWidth, this->height()); break; } } void ExLineEdit::initStyleOption(QStyleOptionFrameV2 *option) const { option->initFrom(this); option->rect = contentsRect(); option->lineWidth = style()->pixelMetric(QStyle::PM_DefaultFrameWidth, option, this); option->midLineWidth = 0; option->state |= QStyle::State_Sunken; if (m_lineEdit->isReadOnly()) option->state |= QStyle::State_ReadOnly; #ifdef QT_KEYPAD_NAVIGATION if (hasEditFocus()) option->state |= QStyle::State_HasEditFocus; #endif option->features = QStyleOptionFrameV2::None; } QSize ExLineEdit::sizeHint() const { m_lineEdit->setFrame(true); QSize size = m_lineEdit->sizeHint(); m_lineEdit->setFrame(false); return size; } void ExLineEdit::focusInEvent(QFocusEvent *event) { m_lineEdit->event(event); QWidget::focusInEvent(event); } void ExLineEdit::focusOutEvent(QFocusEvent *event) { m_lineEdit->event(event); if (m_lineEdit->completer()) { connect(m_lineEdit->completer(), SIGNAL(activated(QString)), m_lineEdit, SLOT(setText(QString))); connect(m_lineEdit->completer(), SIGNAL(highlighted(QString)), m_lineEdit, SLOT(_q_completionHighlighted(QString))); } QWidget::focusOutEvent(event); } void ExLineEdit::keyPressEvent(QKeyEvent *event) { m_lineEdit->event(event); QWidget::keyPressEvent(event); } bool ExLineEdit::event(QEvent *event) { if (event->type() == QEvent::ShortcutOverride) m_lineEdit->event(event); return QWidget::event(event); } void ExLineEdit::paintEvent(QPaintEvent *) { QPainter p(this); QStyleOptionFrameV2 panel; initStyleOption(&panel); style()->drawPrimitive(QStyle::PE_PanelLineEdit, &panel, &p, this); } void ExLineEdit::clear() { m_lineEdit->clear(); } ClearButton::ClearButton(QWidget *parent) : QAbstractButton(parent) { setCursor(Qt::ArrowCursor); setToolTip(tr("Clear")); setVisible(false); setFocusPolicy(Qt::NoFocus); } void ClearButton::paintEvent(QPaintEvent *event) { Q_UNUSED(event); QPainter painter(this); int height = this->height(); painter.setRenderHint(QPainter::Antialiasing, true); QColor color = palette().color(QPalette::Mid); painter.setBrush(isDown() ? palette().color(QPalette::Dark) : palette().color(QPalette::Mid)); painter.setPen(painter.brush().color()); int size = width(); int offset = size / 5; int radius = size - offset * 2; painter.drawEllipse(offset, offset, radius, radius); painter.setPen(palette().color(QPalette::Base)); int border = offset * 2; painter.drawLine(border, border, width() - border, height - border); painter.drawLine(border, height - border, width() - border, border); } void ClearButton::textChanged(const QString &text) { setVisible(!text.isEmpty()); } /* Search icon on the left hand side of the search widget When a menu is set a down arrow appears */ class SearchButton : public QAbstractButton { public: SearchButton(QWidget *parent = 0); void paintEvent(QPaintEvent *event); QMenu *m_menu; protected: void mousePressEvent(QMouseEvent *event); }; SearchButton::SearchButton(QWidget *parent) : QAbstractButton(parent) , m_menu(0) { setObjectName(QLatin1String("SearchButton")); setCursor(Qt::ArrowCursor); setFocusPolicy(Qt::NoFocus); } void SearchButton::mousePressEvent(QMouseEvent *event) { if (m_menu && event->button() == Qt::LeftButton) { QWidget *p = parentWidget(); if (p) { QPoint r = p->mapToGlobal(QPoint(0, p->height())); m_menu->exec(QPoint(r.x() + height() / 2, r.y())); } event->accept(); } QAbstractButton::mousePressEvent(event); } void SearchButton::paintEvent(QPaintEvent *event) { Q_UNUSED(event); QPainterPath myPath; int radius = (height() / 5) * 2; QRect circle(height() / 3 - 1, height() / 4, radius, radius); myPath.addEllipse(circle); myPath.arcMoveTo(circle, 300); QPointF c = myPath.currentPosition(); int diff = height() / 7; myPath.lineTo(qMin(width() - 2, (int)c.x() + diff), c.y() + diff); QPainter painter(this); painter.setRenderHint(QPainter::Antialiasing, true); painter.setPen(QPen(Qt::darkGray, 2)); painter.drawPath(myPath); if (m_menu) { QPainterPath dropPath; dropPath.arcMoveTo(circle, 320); QPointF c = dropPath.currentPosition(); c = QPointF(c.x() + 3.5, c.y() + 0.5); dropPath.moveTo(c); dropPath.lineTo(c.x() + 4, c.y()); dropPath.lineTo(c.x() + 2, c.y() + 2); dropPath.closeSubpath(); painter.setPen(Qt::darkGray); painter.setBrush(Qt::darkGray); painter.setRenderHint(QPainter::Antialiasing, false); painter.drawPath(dropPath); } painter.end(); } /* SearchLineEdit is an enhanced QLineEdit - A Search icon on the left with optional menu - When there is no text and doesn't have focus an "inactive text" is displayed - When there is text a clear button is displayed on the right hand side */ SearchLineEdit::SearchLineEdit(QWidget *parent) : ExLineEdit(parent) , m_searchButton(new SearchButton(this)) { connect(lineEdit(), SIGNAL(textChanged(const QString &)), this, SIGNAL(textChanged(const QString &))); setLeftWidget(m_searchButton); m_inactiveText = tr("Search"); QSizePolicy policy = sizePolicy(); setSizePolicy(QSizePolicy::Preferred, policy.verticalPolicy()); } void SearchLineEdit::paintEvent(QPaintEvent *event) { if (lineEdit()->text().isEmpty() && !hasFocus() && !m_inactiveText.isEmpty()) { ExLineEdit::paintEvent(event); QStyleOptionFrameV2 panel; initStyleOption(&panel); QRect r = style()->subElementRect(QStyle::SE_LineEditContents, &panel, this); QFontMetrics fm = fontMetrics(); int horizontalMargin = lineEdit()->x(); QRect lineRect(horizontalMargin + r.x(), r.y() + (r.height() - fm.height() + 1) / 2, r.width() - 2 * horizontalMargin, fm.height()); QPainter painter(this); painter.setPen(palette().brush(QPalette::Disabled, QPalette::Text).color()); painter.drawText(lineRect, Qt::AlignLeft | Qt::AlignVCenter, m_inactiveText); } else { ExLineEdit::paintEvent(event); } } void SearchLineEdit::resizeEvent(QResizeEvent *event) { updateGeometries(); ExLineEdit::resizeEvent(event); } void SearchLineEdit::updateGeometries() { int menuHeight = height(); int menuWidth = menuHeight + 1; if (!m_searchButton->m_menu) menuWidth = (menuHeight / 5) * 4; m_searchButton->resize(QSize(menuWidth, menuHeight)); } QString SearchLineEdit::inactiveText() const { return m_inactiveText; } void SearchLineEdit::setInactiveText(const QString &text) { m_inactiveText = text; } void SearchLineEdit::setMenu(QMenu *menu) { if (m_searchButton->m_menu) m_searchButton->m_menu->deleteLater(); m_searchButton->m_menu = menu; updateGeometries(); } QMenu *SearchLineEdit::menu() const { if (!m_searchButton->m_menu) { m_searchButton->m_menu = new QMenu(m_searchButton); if (isVisible()) (const_cast(this))->updateGeometries(); } return m_searchButton->m_menu; }