okular/ui/minibar.cpp
2018-08-17 21:05:01 +03:00

599 lines
17 KiB
C++

/***************************************************************************
* Copyright (C) 2005 by Enrico Ros <eros.kde@email.it> *
* Copyright (C) 2006 by Albert Astals Cid <aacid@kde.org> *
* *
* 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) any later version. *
***************************************************************************/
#include "minibar.h"
// qt / kde includes
#include <qapplication.h>
#include <qevent.h>
#include <qframe.h>
#include <qpushbutton.h>
#include <qlabel.h>
#include <qlayout.h>
#include <QToolButton>
#include <qvalidator.h>
#include <qpainter.h>
#include <QIcon>
#include <kicontheme.h>
#include <klineedit.h>
#include <KLocalizedString>
#include <kacceleratormanager.h>
#include <qtoolbar.h>
// local includes
#include "core/document.h"
#include "core/page.h"
// [private widget] a flat qpushbutton that enlights on hover
class HoverButton : public QToolButton
{
Q_OBJECT
public:
HoverButton( QWidget * parent );
};
MiniBarLogic::MiniBarLogic( QObject * parent, Okular::Document * document )
: QObject(parent)
, m_document( document )
{
}
MiniBarLogic::~MiniBarLogic()
{
m_document->removeObserver( this );
}
void MiniBarLogic::addMiniBar( MiniBar * miniBar )
{
m_miniBars.insert( miniBar );
}
void MiniBarLogic::removeMiniBar( MiniBar * miniBar )
{
m_miniBars.remove( miniBar );
}
Okular::Document *MiniBarLogic::document() const
{
return m_document;
}
int MiniBarLogic::currentPage() const
{
return m_document->currentPage();
}
void MiniBarLogic::notifySetup( const QVector< Okular::Page * > & pageVector, int setupFlags )
{
// only process data when document changes
if ( !( setupFlags & Okular::DocumentObserver::DocumentChanged ) )
return;
// if document is closed or has no pages, hide widget
const int pages = pageVector.count();
if ( pages < 1 )
{
foreach ( MiniBar *miniBar, m_miniBars )
{
miniBar->setEnabled( false );
}
return;
}
bool labelsDiffer = false;
foreach(const Okular::Page * page, pageVector)
{
if (!page->label().isEmpty())
{
if (page->label().toInt() != (page->number() + 1))
{
labelsDiffer = true;
}
}
}
const QString pagesString = QString::number( pages );
foreach ( MiniBar *miniBar, m_miniBars )
{
// resize width of widgets
miniBar->resizeForPage( pages );
// update child widgets
miniBar->m_pageLabelEdit->setPageLabels( pageVector );
miniBar->m_pageNumberEdit->setPagesNumber( pages );
miniBar->m_pagesButton->setText( pagesString );
miniBar->m_prevButton->setEnabled( false );
miniBar->m_nextButton->setEnabled( false );
miniBar->m_pageLabelEdit->setVisible( labelsDiffer );
miniBar->m_pageNumberLabel->setVisible( labelsDiffer );
miniBar->m_pageNumberEdit->setVisible( !labelsDiffer );
miniBar->adjustSize();
miniBar->setEnabled( true );
}
}
void MiniBarLogic::notifyCurrentPageChanged( int previousPage, int currentPage )
{
Q_UNUSED( previousPage )
// get current page number
const int pages = m_document->pages();
// if the document is opened and page is changed
if ( pages > 0 )
{
const QString pageNumber = QString::number( currentPage + 1 );
const QString pageLabel = m_document->page( currentPage )->label();
foreach ( MiniBar *miniBar, m_miniBars )
{
// update prev/next button state
miniBar->m_prevButton->setEnabled( currentPage > 0 );
miniBar->m_nextButton->setEnabled( currentPage < ( pages - 1 ) );
// update text on widgets
miniBar->m_pageNumberEdit->setText( pageNumber );
miniBar->m_pageNumberLabel->setText( pageNumber );
miniBar->m_pageLabelEdit->setText( pageLabel );
}
}
}
/** MiniBar **/
MiniBar::MiniBar( QWidget * parent, MiniBarLogic * miniBarLogic )
: QWidget( parent )
, m_miniBarLogic( miniBarLogic )
, m_oldToobarParent( nullptr )
{
setObjectName( QStringLiteral( "miniBar" ) );
m_miniBarLogic->addMiniBar( this );
QHBoxLayout * horLayout = new QHBoxLayout( this );
horLayout->setMargin( 0 );
horLayout->setSpacing( 3 );
QSize buttonSize( KIconLoader::SizeSmallMedium, KIconLoader::SizeSmallMedium );
// bottom: left prev_page button
m_prevButton = new HoverButton( this );
m_prevButton->setIcon( QIcon::fromTheme( layoutDirection() == Qt::RightToLeft ? QStringLiteral("arrow-right") : QStringLiteral("arrow-left") ) );
m_prevButton->setIconSize( buttonSize );
horLayout->addWidget( m_prevButton );
// bottom: left lineEdit (current page box)
m_pageNumberEdit = new PageNumberEdit( this );
horLayout->addWidget( m_pageNumberEdit );
m_pageNumberEdit->installEventFilter( this );
// bottom: left labelWidget (current page label)
m_pageLabelEdit = new PageLabelEdit( this );
horLayout->addWidget(m_pageLabelEdit);
m_pageLabelEdit->installEventFilter( this );
// bottom: left labelWidget (current page label)
m_pageNumberLabel = new QLabel( this );
m_pageNumberLabel->setAlignment( Qt::AlignCenter );
horLayout->addWidget(m_pageNumberLabel);
// bottom: central 'of' label
horLayout->addSpacing(5);
horLayout->addWidget( new QLabel( i18nc( "Layouted like: '5 [pages] of 10'", "of" ), this ) );
// bottom: right button
m_pagesButton = new HoverButton( this );
horLayout->addWidget( m_pagesButton );
// bottom: right next_page button
m_nextButton = new HoverButton( this );
m_nextButton->setIcon( QIcon::fromTheme( layoutDirection() == Qt::RightToLeft ? QStringLiteral("arrow-left") : QStringLiteral("arrow-right") ) );
m_nextButton->setIconSize( buttonSize );
horLayout->addWidget( m_nextButton );
QSizePolicy sp = sizePolicy();
sp.setHorizontalPolicy( QSizePolicy::Fixed );
sp.setVerticalPolicy( QSizePolicy::Fixed );
setSizePolicy( sp );
// resize width of widgets
resizeForPage( 0 );
// connect signals from child widgets to internal handlers / signals bouncers
connect( m_pageNumberEdit, SIGNAL(returnPressed()), this, SLOT(slotChangePage()) );
connect( m_pageLabelEdit, SIGNAL(pageNumberChosen(int)), this, SLOT(slotChangePage(int)) );
connect( m_pagesButton, &QAbstractButton::clicked, this, &MiniBar::gotoPage );
connect( m_prevButton, &QAbstractButton::clicked, this, &MiniBar::prevPage );
connect( m_nextButton, &QAbstractButton::clicked, this, &MiniBar::nextPage );
adjustSize();
// widget starts disabled (will be enabled after opening a document)
setEnabled( false );
}
MiniBar::~MiniBar()
{
m_miniBarLogic->removeMiniBar( this );
}
void MiniBar::changeEvent( QEvent * event )
{
if ( event->type() == QEvent::ParentChange )
{
QToolBar *tb = dynamic_cast<QToolBar*>( parent() );
if ( tb != m_oldToobarParent )
{
if ( m_oldToobarParent )
{
disconnect( m_oldToobarParent, &QToolBar::iconSizeChanged, this, &MiniBar::slotToolBarIconSizeChanged);
}
m_oldToobarParent = tb;
if ( tb )
{
connect( tb, &QToolBar::iconSizeChanged, this, &MiniBar::slotToolBarIconSizeChanged);
slotToolBarIconSizeChanged();
}
}
}
}
bool MiniBar::eventFilter( QObject *target, QEvent *event )
{
if ( target == m_pageNumberEdit || target == m_pageLabelEdit )
{
if ( event->type() == QEvent::KeyPress )
{
QKeyEvent *keyEvent = static_cast<QKeyEvent *>( event );
int key = keyEvent->key();
if ( key == Qt::Key_PageUp || key == Qt::Key_PageDown || key == Qt::Key_Up || key == Qt::Key_Down )
{
emit forwardKeyPressEvent( keyEvent );
return true;
}
}
}
return false;
}
void MiniBar::slotChangePage()
{
// get text from the lineEdit
const QString pageNumber = m_pageNumberEdit->text();
// convert it to page number and go to that page
bool ok;
int number = pageNumber.toInt( &ok ) - 1;
if ( ok && number >= 0 && number < (int)m_miniBarLogic->document()->pages() &&
number != m_miniBarLogic->currentPage() )
{
slotChangePage( number );
}
}
void MiniBar::slotChangePage( int pageNumber )
{
m_miniBarLogic->document()->setViewportPage( pageNumber );
m_pageNumberEdit->clearFocus();
m_pageLabelEdit->clearFocus();
}
void MiniBar::slotEmitNextPage()
{
// emit signal
nextPage();
}
void MiniBar::slotEmitPrevPage()
{
// emit signal
prevPage();
}
void MiniBar::slotToolBarIconSizeChanged()
{
const QSize buttonSize = m_oldToobarParent->iconSize();
m_prevButton->setIconSize( buttonSize );
m_nextButton->setIconSize( buttonSize );
}
void MiniBar::resizeForPage( int pages )
{
int numberWidth = 10 + fontMetrics().width( QString::number( pages ) );
m_pageNumberEdit->setMinimumWidth( numberWidth );
m_pageNumberEdit->setMaximumWidth( 2 * numberWidth );
m_pageLabelEdit->setMinimumWidth( numberWidth );
m_pageLabelEdit->setMaximumWidth( 2 * numberWidth );
m_pageNumberLabel->setMinimumWidth( numberWidth );
m_pageNumberLabel->setMaximumWidth( 2 * numberWidth );
m_pagesButton->setMinimumWidth( numberWidth );
m_pagesButton->setMaximumWidth( 2 * numberWidth );
}
/** ProgressWidget **/
ProgressWidget::ProgressWidget( QWidget * parent, Okular::Document * document )
: QWidget( parent ), m_document( document ),
m_progressPercentage( -1 )
{
setObjectName( QStringLiteral( "progress" ) );
setAttribute( Qt::WA_OpaquePaintEvent, true );
setFixedHeight( 4 );
setMouseTracking( true );
}
ProgressWidget::~ProgressWidget()
{
m_document->removeObserver( this );
}
void ProgressWidget::notifyCurrentPageChanged( int previousPage, int currentPage )
{
Q_UNUSED( previousPage )
// get current page number
int pages = m_document->pages();
// if the document is opened and page is changed
if ( pages > 0 )
{
// update percentage
const float percentage = pages < 2 ? 1.0 : (float)currentPage / (float)(pages - 1);
setProgress( percentage );
}
}
void ProgressWidget::setProgress( float percentage )
{
m_progressPercentage = percentage;
update();
}
void ProgressWidget::slotGotoNormalizedPage( float index )
{
// figure out page number and go to that page
int number = (int)( index * (float)m_document->pages() );
if ( number >= 0 && number < (int)m_document->pages() &&
number != (int)m_document->currentPage() )
m_document->setViewportPage( number );
}
void ProgressWidget::mouseMoveEvent( QMouseEvent * e )
{
if ( ( QApplication::mouseButtons() & Qt::LeftButton ) && width() > 0 )
slotGotoNormalizedPage( (float)( QApplication::isRightToLeft() ? width() - e->x() : e->x() ) / (float)width() );
}
void ProgressWidget::mousePressEvent( QMouseEvent * e )
{
if ( e->button() == Qt::LeftButton && width() > 0 )
slotGotoNormalizedPage( (float)( QApplication::isRightToLeft() ? width() - e->x() : e->x() ) / (float)width() );
}
void ProgressWidget::wheelEvent( QWheelEvent * e )
{
if ( e->delta() > 0 )
emit nextPage();
else
emit prevPage();
}
void ProgressWidget::paintEvent( QPaintEvent * e )
{
QPainter p( this );
if ( m_progressPercentage < 0.0 )
{
p.fillRect( rect(), palette().color( QPalette::Active, QPalette::HighlightedText ) );
return;
}
// find out the 'fill' and the 'clear' rectangles
int w = width(),
h = height(),
l = (int)( (float)w * m_progressPercentage );
QRect cRect = ( QApplication::isRightToLeft() ? QRect( 0, 0, w - l, h ) : QRect( l, 0, w - l, h ) ).intersected( e->rect() );
QRect fRect = ( QApplication::isRightToLeft() ? QRect( w - l, 0, l, h ) : QRect( 0, 0, l, h ) ).intersected( e->rect() );
QPalette pal = palette();
// paint clear rect
if ( cRect.isValid() )
p.fillRect( cRect, pal.color( QPalette::Active, QPalette::HighlightedText ) );
// draw a frame-like outline
//p.setPen( palette().active().mid() );
//p.drawRect( 0,0, w, h );
// paint fill rect
if ( fRect.isValid() )
p.fillRect( fRect, pal.color( QPalette::Active, QPalette::Highlight ) );
if ( l && l != w )
{
p.setPen( pal.color( QPalette::Active, QPalette::Highlight ).dark( 120 ) );
int delta = QApplication::isRightToLeft() ? w - l : l;
p.drawLine( delta, 0, delta, h );
}
}
/** PageLabelEdit **/
PageLabelEdit::PageLabelEdit( MiniBar * parent )
: PagesEdit( parent )
{
setVisible( false );
connect( this, SIGNAL(returnPressed()), this, SLOT(pageChosen()) );
}
void PageLabelEdit::setText( const QString & newText )
{
m_lastLabel = newText;
PagesEdit::setText( newText );
}
void PageLabelEdit::setPageLabels( const QVector<Okular::Page *> &pageVector )
{
m_labelPageMap.clear();
completionObject()->clear();
foreach(const Okular::Page * page, pageVector)
{
if ( !page->label().isEmpty() )
{
m_labelPageMap.insert( page->label(), page->number() );
bool ok;
page->label().toInt( &ok );
if ( !ok )
{
// Only add to the completion objects labels that are not numbers
completionObject()->addItem( page->label() );
}
}
}
}
void PageLabelEdit::pageChosen()
{
const QString newInput = text();
const int pageNumber = m_labelPageMap.value( newInput, -1 );
if (pageNumber != -1)
{
emit pageNumberChosen( pageNumber );
}
else
{
setText( m_lastLabel );
}
}
/** PageNumberEdit **/
PageNumberEdit::PageNumberEdit( MiniBar * miniBar )
: PagesEdit( miniBar )
{
// use an integer validator
m_validator = new QIntValidator( 1, 1, this );
setValidator( m_validator );
}
void PageNumberEdit::setPagesNumber( int pages )
{
m_validator->setTop( pages );
}
/** PagesEdit **/
PagesEdit::PagesEdit( MiniBar * parent )
: KLineEdit( parent ), m_miniBar( parent ), m_eatClick( false )
{
// customize text properties
setAlignment( Qt::AlignCenter );
// send a focus out event
QFocusEvent fe( QEvent::FocusOut );
QApplication::sendEvent( this, &fe );
connect( qApp, &QGuiApplication::paletteChanged, this, &PagesEdit::updatePalette );
}
void PagesEdit::setText( const QString & newText )
{
// call default handler if hasn't focus
if ( !hasFocus() )
{
KLineEdit::setText( newText );
}
// else preserve existing selection
else
{
// save selection and adapt it to the new text length
int selectionLength = selectedText().length();
const bool allSelected = ( selectionLength == text().length() );
if ( allSelected )
{
KLineEdit::setText( newText );
selectAll();
}
else
{
int newSelectionStart = newText.length() - text().length() + selectionStart();
if ( newSelectionStart < 0 )
{
// the new text is shorter than the old one, and the front part, which is "cut off", is selected
// shorten the selection accordingly
selectionLength += newSelectionStart;
newSelectionStart = 0;
}
KLineEdit::setText( newText );
setSelection( newSelectionStart, selectionLength );
}
}
}
void PagesEdit::updatePalette()
{
QPalette pal;
if ( hasFocus() )
pal.setColor( QPalette::Active, QPalette::Base, QApplication::palette().color( QPalette::Active, QPalette::Base ) );
else
pal.setColor( QPalette::Base, QApplication::palette().color( QPalette::Base ).dark( 102 ) );
setPalette( pal );
}
void PagesEdit::focusInEvent( QFocusEvent * e )
{
// select all text
selectAll();
if ( e->reason() == Qt::MouseFocusReason )
m_eatClick = true;
// change background color to the default 'edit' color
updatePalette();
// call default handler
KLineEdit::focusInEvent( e );
}
void PagesEdit::focusOutEvent( QFocusEvent * e )
{
// change background color to a dark tone
updatePalette();
// call default handler
KLineEdit::focusOutEvent( e );
}
void PagesEdit::mousePressEvent( QMouseEvent * e )
{
// if this click got the focus in, don't process the event
if ( !m_eatClick )
KLineEdit::mousePressEvent( e );
m_eatClick = false;
}
void PagesEdit::wheelEvent( QWheelEvent * e )
{
if ( e->delta() > 0 )
m_miniBar->slotEmitNextPage();
else
m_miniBar->slotEmitPrevPage();
}
/** HoverButton **/
HoverButton::HoverButton( QWidget * parent )
: QToolButton( parent )
{
setAutoRaise(true);
setFocusPolicy(Qt::NoFocus);
setToolButtonStyle(Qt::ToolButtonIconOnly);
KAcceleratorManager::setNoAccel( this );
}
#include "minibar.moc"
/* kate: replace-tabs on; indent-width 4; */