/* ============================================================ * * This file is a part of the rekonq project * * Copyright (C) 2009 by Andrea Diamantini <adjam7 at gmail dot com> * * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of * the License or (at your option) version 3 or any later version * accepted by the membership of KDE e.V. (or its successor approved * by the membership of KDE e.V.), which shall act as a proxy * defined in Section 14 of version 3 of the license. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. * * ============================================================ */ // Self Includes #include "completionwidget.h" #include "completionwidget.moc" // Auto Includes #include "rekonq.h" // Local Includes #include "application.h" #include "urlresolver.h" #include "searchengine.h" // KDE Includes #include <KGlobalSettings> #include <KDebug> #include <KUrl> // Qt Includes #include <QPoint> #include <QSize> #include <QVBoxLayout> #include <QString> #include <QEvent> #include <QKeyEvent> CompletionWidget::CompletionWidget(QWidget *parent) : QFrame(parent, Qt::ToolTip) , _parent(parent) , _currentIndex(-1) , _searchEngine( SearchEngine::defaultEngine() ) { setFrameStyle(QFrame::Panel); setLayoutDirection(Qt::LeftToRight); QVBoxLayout *layout = new QVBoxLayout; layout->setMargin(0); layout->setSpacing(0); setLayout(layout); } void CompletionWidget::insertSearchList(const UrlSearchList &list, const QString& text) { _list = list; int i = 0; foreach(UrlSearchItem item, _list) { ListItem *suggestion = ListItemFactory::create(item, text, this); suggestion->setBackgroundRole(i%2 ? QPalette::AlternateBase : QPalette::Base); connect(suggestion, SIGNAL(itemClicked(ListItem *, Qt::MouseButton)), this, SLOT(itemChosen(ListItem *, Qt::MouseButton))); connect(this, SIGNAL(nextItemSubChoice()), suggestion, SLOT(nextItemSubChoice())); suggestion->setObjectName( QString::number(i++) ); layout()->addWidget( suggestion ); } } void CompletionWidget::sizeAndPosition() { setFixedWidth( _parent->width() ); adjustSize(); // position QPoint p = _parent->mapToGlobal( QPoint(0,0) ); move(p.x(), p.y() + _parent->height()); } void CompletionWidget::popup() { down(); sizeAndPosition(); if (!isVisible()) show(); } void CompletionWidget::up() { // deactivate previous if(_currentIndex != -1) { ListItem *widget = findChild<ListItem *>( QString::number(_currentIndex) ); widget->deactivate(); } if(_currentIndex > 0) _currentIndex--; else _currentIndex=layout()->count()-1; // activate "new" current ListItem *widget = findChild<ListItem *>( QString::number(_currentIndex) ); widget->activate(); } void CompletionWidget::down() { // deactivate previous if(_currentIndex != -1) { ListItem *widget = findChild<ListItem *>( QString::number(_currentIndex) ); widget->deactivate(); } if(_currentIndex < _list.count() -1) _currentIndex++; else _currentIndex=0; // activate "new" current ListItem *widget = findChild<ListItem *>( QString::number(_currentIndex) ); widget->activate(); } void CompletionWidget::clear() { QLayoutItem *child; while ((child = layout()->takeAt(0)) != 0) { delete child->widget(); delete child; } _currentIndex = -1; } bool CompletionWidget::eventFilter( QObject *o, QEvent *e ) { int type = e->type(); QWidget *wid = qobject_cast<QWidget*>(o); if (o == this) { return false; } //hide conditions of the CompletionWidget if (wid && ((wid == _parent && (type == QEvent::Move || type == QEvent::Resize)) || ((wid->windowFlags() & Qt::Window) && (type == QEvent::Move || type == QEvent::Hide || type == QEvent::WindowDeactivate) && wid == _parent->window()) || (type == QEvent::MouseButtonPress && !isAncestorOf(wid))) ) { hide(); return false; } //actions on the CompletionWidget if (wid && wid->isAncestorOf(_parent) && isVisible()) { ListItem *child; if ( type == QEvent::KeyPress ) { QKeyEvent *ev = static_cast<QKeyEvent *>( e ); switch ( ev->key() ) { case Qt::Key_Up: case Qt::Key_Backtab: if (ev->modifiers() == Qt::NoButton || (ev->modifiers() & Qt::ShiftModifier)) { up(); ev->accept(); return true; } break; case Qt::Key_Down: case Qt::Key_Tab: if (ev->modifiers() == Qt::NoButton) { down(); ev->accept(); return true; } if (ev->modifiers() & Qt::ControlModifier) { emit nextItemSubChoice(); ev->accept(); return true; } break; case Qt::Key_Enter: case Qt::Key_Return: child = findChild<ListItem *>( QString::number(_currentIndex) ); emit chosenUrl( child->url(), Rekonq::CurrentTab); ev->accept(); hide(); return true; case Qt::Key_Escape: hide(); return true; } } } return QFrame::eventFilter(o,e); } void CompletionWidget::setVisible( bool visible ) { if (visible) { Application::instance()->installEventFilter(this); } else { Application::instance()->removeEventFilter(this); } QFrame::setVisible(visible); } void CompletionWidget::itemChosen(ListItem *item, Qt::MouseButton button) { if(button == Qt::MidButton) emit chosenUrl( item->url(), Rekonq::NewCurrentTab); else emit chosenUrl( item->url(), Rekonq::CurrentTab); hide(); } void CompletionWidget::suggestUrls(const QString &text) { QWidget *w = qobject_cast<QWidget *>(parent()); if(!w->hasFocus()) return; if(text.isEmpty()) { hide(); return; } UrlResolver res(text); UrlSearchList list = res.orderedSearchItems(); if(list.count() > 0) { clear(); insertSearchList(list, text); popup(); } }