mirror of
https://bitbucket.org/smil3y/kde-playground.git
synced 2025-02-24 02:42:51 +00:00
591 lines
15 KiB
C++
591 lines
15 KiB
C++
/*
|
|
KNode, the KDE newsreader
|
|
Copyright (c) 1999-2005 the KNode authors.
|
|
See file AUTHORS for details
|
|
|
|
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.
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software Foundation,
|
|
Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, US
|
|
*/
|
|
|
|
#include <QCursor>
|
|
#include <q3header.h>
|
|
#include <q3stylesheet.h>
|
|
#include <QTimer>
|
|
|
|
#include <QKeyEvent>
|
|
#include <QEvent>
|
|
#include <QMouseEvent>
|
|
|
|
#include <klocale.h>
|
|
#include <kdebug.h>
|
|
#include <kmenu.h>
|
|
|
|
#include "knglobals.h"
|
|
#include "headerview.h"
|
|
#include "knhdrviewitem.h"
|
|
#include "kngroupmanager.h"
|
|
#include "knarticle.h"
|
|
#include "knarticlemanager.h"
|
|
#include "knmainwidget.h"
|
|
#include "settings.h"
|
|
|
|
|
|
KNHeaderView::KNHeaderView( QWidget *parent ) :
|
|
K3ListView( parent ),
|
|
mSortCol( -1 ),
|
|
mSortAsc( true ),
|
|
mSortByThreadChangeDate( false ),
|
|
mDelayedCenter( -1 ),
|
|
mActiveItem( 0 ),
|
|
mShowingFolder( false ),
|
|
mInitDone( false )
|
|
{
|
|
mPaintInfo.subCol = addColumn( i18n("Subject"), 310 );
|
|
mPaintInfo.senderCol = addColumn( i18n("From"), 115 );
|
|
mPaintInfo.scoreCol = addColumn( i18n("Score"), 42 );
|
|
mPaintInfo.sizeCol = addColumn( i18n("Lines"), 42 );
|
|
mPaintInfo.dateCol = addColumn( i18n("Date"), 102 );
|
|
|
|
setDropVisualizer( false );
|
|
setDropHighlighter( false );
|
|
setItemsRenameable( false );
|
|
setItemsMovable( false );
|
|
setAcceptDrops( false );
|
|
setDragEnabled( true );
|
|
setAllColumnsShowFocus( true );
|
|
setSelectionMode( Q3ListView::Extended );
|
|
setShowSortIndicator( true );
|
|
setShadeSortColumn ( true );
|
|
setRootIsDecorated( true );
|
|
setSorting( mPaintInfo.dateCol );
|
|
header()->setMovingEnabled( true );
|
|
setColumnAlignment( mPaintInfo.sizeCol, Qt::AlignRight );
|
|
setColumnAlignment( mPaintInfo.scoreCol, Qt::AlignRight );
|
|
|
|
// due to our own column text squeezing we need to repaint on column resizing
|
|
disconnect( header(), SIGNAL(sizeChange(int,int,int)) );
|
|
connect( header(), SIGNAL(sizeChange(int,int,int)),
|
|
SLOT(slotSizeChanged(int,int,int)) );
|
|
|
|
// column selection RMB menu
|
|
mPopup = new KMenu( this );
|
|
mPopup->addTitle( i18n("View Columns") );
|
|
mPopup->setCheckable( true );
|
|
mPopup->insertItem( i18n("Line Count"), KPaintInfo::COL_SIZE );
|
|
mPopup->insertItem( i18n("Score"), KPaintInfo::COL_SCORE );
|
|
|
|
connect( mPopup, SIGNAL(activated(int)), this, SLOT(toggleColumn(int)) );
|
|
|
|
// connect to the article manager
|
|
connect( knGlobals.articleManager(), SIGNAL(aboutToShowGroup()), SLOT(prepareForGroup()) );
|
|
connect( knGlobals.articleManager(), SIGNAL(aboutToShowFolder()), SLOT(prepareForFolder()) );
|
|
|
|
#ifdef __GNUC__
|
|
#warning Port me!
|
|
#endif
|
|
// new KNHeaderViewToolTip( this );
|
|
|
|
installEventFilter( this );
|
|
}
|
|
|
|
|
|
KNHeaderView::~KNHeaderView()
|
|
{
|
|
// ### crash because KNConfigManager is already deleted here
|
|
// writeConfig();
|
|
}
|
|
|
|
|
|
void KNHeaderView::readConfig()
|
|
{
|
|
if ( !mInitDone ) {
|
|
KConfigGroup conf(knGlobals.config(), "HeaderView" );
|
|
mSortByThreadChangeDate = conf.readEntry( "sortByThreadChangeDate", false );
|
|
restoreLayout( knGlobals.config(), "HeaderView" );
|
|
mInitDone = true;
|
|
}
|
|
|
|
toggleColumn( KPaintInfo::COL_SIZE, knGlobals.settings()->showLines() );
|
|
if ( !mShowingFolder ) // score column is always hidden when showing a folder
|
|
toggleColumn( KPaintInfo::COL_SCORE, knGlobals.settings()->showScore() );
|
|
|
|
mDateFormatter.setCustomFormat( knGlobals.settings()->customDateFormat() );
|
|
mDateFormatter.setFormat( knGlobals.settings()->dateFormat() );
|
|
|
|
QPalette p = palette();
|
|
p.setColor( QPalette::Base, knGlobals.settings()->backgroundColor() );
|
|
p.setColor( QPalette::Text, knGlobals.settings()->textColor() );
|
|
setPalette( p );
|
|
setAlternateBackground( knGlobals.settings()->alternateBackgroundColor() );
|
|
setFont( knGlobals.settings()->articleListFont() );
|
|
}
|
|
|
|
|
|
void KNHeaderView::writeConfig()
|
|
{
|
|
KConfigGroup conf(knGlobals.config(), "HeaderView" );
|
|
conf.writeEntry( "sortByThreadChangeDate", mSortByThreadChangeDate );
|
|
saveLayout( knGlobals.config(), "HeaderView" );
|
|
|
|
knGlobals.settings()->setShowLines( mPaintInfo.showSize );
|
|
if ( !mShowingFolder ) // score column is always hidden when showing a folder
|
|
knGlobals.settings()->setShowScore( mPaintInfo.showScore );
|
|
}
|
|
|
|
|
|
void KNHeaderView::setActive( Q3ListViewItem *i )
|
|
{
|
|
KNHdrViewItem *item = static_cast<KNHdrViewItem*>( i );
|
|
|
|
if ( !item || item->isActive() )
|
|
return;
|
|
|
|
if ( mActiveItem ) {
|
|
mActiveItem->setActive( false );
|
|
repaintItem( mActiveItem );
|
|
mActiveItem = 0;
|
|
}
|
|
|
|
item->setActive( true );
|
|
setSelected( item, true );
|
|
setCurrentItem( i );
|
|
ensureItemVisibleWithMargin( i );
|
|
mActiveItem = item;
|
|
emit( itemSelected(item) );
|
|
}
|
|
|
|
|
|
void KNHeaderView::clear()
|
|
{
|
|
mActiveItem = 0;
|
|
Q3ListView::clear();
|
|
}
|
|
|
|
|
|
void KNHeaderView::ensureItemVisibleWithMargin( const Q3ListViewItem *i )
|
|
{
|
|
if ( !i )
|
|
return;
|
|
|
|
Q3ListViewItem *parent = i->parent();
|
|
while ( parent ) {
|
|
if ( !parent->isOpen() )
|
|
parent->setOpen( true );
|
|
parent = parent->parent();
|
|
}
|
|
|
|
mDelayedCenter = -1;
|
|
int y = itemPos( i );
|
|
int h = i->height();
|
|
|
|
if ( knGlobals.settings()->smartScrolling() &&
|
|
((y + h + 5) >= (contentsY() + visibleHeight()) ||
|
|
(y - 5 < contentsY())) )
|
|
{
|
|
ensureVisible( contentsX(), y + h/2, 0, h/2 );
|
|
mDelayedCenter = y + h/2;
|
|
QTimer::singleShot( 300, this, SLOT(slotCenterDelayed()) );
|
|
} else {
|
|
ensureVisible( contentsX(), y + h/2, 0, h/2 );
|
|
}
|
|
}
|
|
|
|
|
|
void KNHeaderView::slotCenterDelayed()
|
|
{
|
|
if ( mDelayedCenter != -1 )
|
|
ensureVisible( contentsX(), mDelayedCenter, 0, visibleHeight() / 2 );
|
|
}
|
|
|
|
|
|
void KNHeaderView::setSorting( int column, bool ascending )
|
|
{
|
|
if ( column == mSortCol ) {
|
|
mSortAsc = ascending;
|
|
if ( mInitDone && column == mPaintInfo.dateCol && ascending )
|
|
mSortByThreadChangeDate = !mSortByThreadChangeDate;
|
|
} else {
|
|
mSortCol = column;
|
|
emit sortingChanged( column );
|
|
}
|
|
|
|
K3ListView::setSorting( column, ascending );
|
|
|
|
if ( currentItem() )
|
|
ensureItemVisible( currentItem() );
|
|
|
|
if ( mSortByThreadChangeDate )
|
|
setColumnText( mPaintInfo.dateCol , i18n("Date (thread changed)") );
|
|
else
|
|
setColumnText( mPaintInfo.dateCol, i18n("Date") );
|
|
}
|
|
|
|
|
|
void KNHeaderView::nextArticle()
|
|
{
|
|
KNHdrViewItem *it = static_cast<KNHdrViewItem*>( currentItem() );
|
|
|
|
if (it) {
|
|
if (it->isActive()) { // take current article, if not selected
|
|
if (it->isExpandable())
|
|
it->setOpen(true);
|
|
it = static_cast<KNHdrViewItem*>(it->itemBelow());
|
|
}
|
|
} else
|
|
it = static_cast<KNHdrViewItem*>( firstChild() );
|
|
|
|
if(it) {
|
|
clearSelection();
|
|
setActive( it );
|
|
setSelectionAnchor( currentItem() );
|
|
}
|
|
}
|
|
|
|
|
|
void KNHeaderView::prevArticle()
|
|
{
|
|
KNHdrViewItem *it = static_cast<KNHdrViewItem*>( currentItem() );
|
|
|
|
if (it && it->isActive()) { // take current article, if not selected
|
|
it = static_cast<KNHdrViewItem*>(it->itemAbove());
|
|
clearSelection();
|
|
setActive( it );
|
|
setSelectionAnchor( currentItem() );
|
|
}
|
|
}
|
|
|
|
|
|
void KNHeaderView::incCurrentArticle()
|
|
{
|
|
Q3ListViewItem *lvi = currentItem();
|
|
if ( lvi && lvi->isExpandable() )
|
|
lvi->setOpen( true );
|
|
if ( lvi && lvi->itemBelow() ) {
|
|
setCurrentItem( lvi->itemBelow() );
|
|
ensureItemVisible( currentItem() );
|
|
setFocus();
|
|
}
|
|
}
|
|
|
|
void KNHeaderView::decCurrentArticle()
|
|
{
|
|
Q3ListViewItem *lvi = currentItem();
|
|
if ( lvi && lvi->itemAbove() ) {
|
|
if ( lvi->itemAbove()->isExpandable() )
|
|
lvi->itemAbove()->setOpen( true );
|
|
setCurrentItem( lvi->itemAbove() );
|
|
ensureItemVisible( currentItem() );
|
|
setFocus();
|
|
}
|
|
}
|
|
|
|
|
|
void KNHeaderView::selectCurrentArticle()
|
|
{
|
|
clearSelection();
|
|
setActive( currentItem() );
|
|
}
|
|
|
|
|
|
bool KNHeaderView::nextUnreadArticle()
|
|
{
|
|
if ( !knGlobals.groupManager()->currentGroup() )
|
|
return false;
|
|
|
|
KNHdrViewItem *next, *current;
|
|
KNRemoteArticle::Ptr art;
|
|
|
|
current = static_cast<KNHdrViewItem*>( currentItem() );
|
|
if ( !current )
|
|
current = static_cast<KNHdrViewItem*>( firstChild() );
|
|
|
|
if(!current)
|
|
return false;
|
|
|
|
art = boost::static_pointer_cast<KNRemoteArticle>( current->art );
|
|
|
|
if ( !current->isActive() && !art->isRead() ) // take current article, if unread & not selected
|
|
next = current;
|
|
else {
|
|
if ( current->isExpandable() && art->hasUnreadFollowUps() && !current->isOpen() )
|
|
setOpen( current, true );
|
|
next = static_cast<KNHdrViewItem*>( current->itemBelow() );
|
|
}
|
|
|
|
while ( next ) {
|
|
art = boost::static_pointer_cast<KNRemoteArticle>( next->art );
|
|
if ( !art->isRead() )
|
|
break;
|
|
else {
|
|
if ( next->isExpandable() && art->hasUnreadFollowUps() && !next->isOpen() )
|
|
setOpen( next, true );
|
|
next = static_cast<KNHdrViewItem*>( next->itemBelow() );
|
|
}
|
|
}
|
|
|
|
if ( next ) {
|
|
clearSelection();
|
|
setActive( next );
|
|
setSelectionAnchor( currentItem() );
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|
|
bool KNHeaderView::nextUnreadThread()
|
|
{
|
|
KNHdrViewItem *next, *current;
|
|
KNRemoteArticle::Ptr art;
|
|
|
|
if ( !knGlobals.groupManager()->currentGroup() )
|
|
return false;
|
|
|
|
current = static_cast<KNHdrViewItem*>( currentItem() );
|
|
if ( !current )
|
|
current = static_cast<KNHdrViewItem*>( firstChild() );
|
|
|
|
if ( !current )
|
|
return false;
|
|
|
|
art = boost::static_pointer_cast<KNRemoteArticle>( current->art );
|
|
|
|
if ( current->depth() == 0 && !current->isActive() && (!art->isRead() || art->hasUnreadFollowUps()) )
|
|
next = current; // take current article, if unread & not selected
|
|
else
|
|
next = static_cast<KNHdrViewItem*>( current->itemBelow() );
|
|
|
|
while ( next ) {
|
|
art = boost::static_pointer_cast<KNRemoteArticle>( next->art );
|
|
|
|
if ( next->depth() == 0 ) {
|
|
if ( !art->isRead() || art->hasUnreadFollowUps() )
|
|
break;
|
|
}
|
|
next = static_cast<KNHdrViewItem*>( next->itemBelow() );
|
|
}
|
|
|
|
if ( next ) {
|
|
setCurrentItem( next );
|
|
if ( art->isRead() )
|
|
nextUnreadArticle();
|
|
else {
|
|
clearSelection();
|
|
setActive( next );
|
|
setSelectionAnchor( currentItem() );
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|
|
void KNHeaderView::toggleColumn( int column, int mode )
|
|
{
|
|
bool *show = 0;
|
|
int *col = 0;
|
|
int width = 0;
|
|
|
|
switch ( static_cast<KPaintInfo::ColumnIds>( column ) )
|
|
{
|
|
case KPaintInfo::COL_SIZE:
|
|
show = &mPaintInfo.showSize;
|
|
col = &mPaintInfo.sizeCol;
|
|
width = 42;
|
|
break;
|
|
case KPaintInfo::COL_SCORE:
|
|
show = &mPaintInfo.showScore;
|
|
col = &mPaintInfo.scoreCol;
|
|
width = 42;
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
|
|
if ( mode == -1 )
|
|
*show = !*show;
|
|
else
|
|
*show = mode;
|
|
|
|
mPopup->setItemChecked( column, *show );
|
|
|
|
if (*show) {
|
|
header()->setResizeEnabled( true, *col );
|
|
setColumnWidth( *col, width );
|
|
}
|
|
else {
|
|
header()->setResizeEnabled( false, *col );
|
|
header()->setStretchEnabled( false, *col );
|
|
hideColumn( *col );
|
|
}
|
|
|
|
if ( mode == -1 ) // save config when toggled
|
|
writeConfig();
|
|
}
|
|
|
|
|
|
void KNHeaderView::prepareForGroup()
|
|
{
|
|
mShowingFolder = false;
|
|
header()->setLabel( mPaintInfo.senderCol, i18n("From") );
|
|
toggleColumn( KPaintInfo::COL_SCORE, knGlobals.settings()->showScore() );
|
|
}
|
|
|
|
|
|
void KNHeaderView::prepareForFolder()
|
|
{
|
|
mShowingFolder = true;
|
|
header()->setLabel( mPaintInfo.senderCol, i18n("Newsgroups / To") );
|
|
toggleColumn( KPaintInfo::COL_SCORE, false ); // local folders have no score
|
|
}
|
|
|
|
|
|
bool KNHeaderView::event( QEvent *e )
|
|
{
|
|
// we don't want to have the alternate list background restored
|
|
// to the system defaults!
|
|
if (e->type() == QEvent::ApplicationPaletteChange)
|
|
return Q3ListView::event(e);
|
|
else
|
|
return K3ListView::event(e);
|
|
}
|
|
|
|
void KNHeaderView::contentsMousePressEvent( QMouseEvent *e )
|
|
{
|
|
if (!e) return;
|
|
|
|
bool selectMode=(( e->modifiers() & Qt::ShiftModifier ) || ( e->modifiers() & Qt::ControlModifier ));
|
|
|
|
QPoint vp = contentsToViewport(e->pos());
|
|
Q3ListViewItem *i = itemAt(vp);
|
|
|
|
K3ListView::contentsMousePressEvent( e );
|
|
|
|
if ( i ) {
|
|
int decoLeft = header()->sectionPos( 0 ) +
|
|
treeStepSize() * ( (i->depth() - 1) + ( rootIsDecorated() ? 1 : 0) );
|
|
int decoRight = qMin( decoLeft + treeStepSize() + itemMargin(),
|
|
header()->sectionPos( 0 ) + header()->sectionSize( 0 ) );
|
|
bool rootDecoClicked = vp.x() > decoLeft && vp.x() < decoRight;
|
|
|
|
if( !selectMode && i->isSelected() && !rootDecoClicked )
|
|
setActive( i );
|
|
}
|
|
}
|
|
|
|
|
|
void KNHeaderView::contentsMouseDoubleClickEvent( QMouseEvent *e )
|
|
{
|
|
if (!e) return;
|
|
|
|
Q3ListViewItem *i = itemAt( contentsToViewport(e->pos()) );
|
|
if (i) {
|
|
emit doubleClick( i );
|
|
return;
|
|
}
|
|
|
|
K3ListView::contentsMouseDoubleClickEvent( e );
|
|
}
|
|
|
|
|
|
void KNHeaderView::keyPressEvent(QKeyEvent *e)
|
|
{
|
|
if (!e) return;
|
|
|
|
Q3ListViewItem *i = currentItem();
|
|
|
|
switch(e->key()) {
|
|
case Qt::Key_Space:
|
|
case Qt::Key_Backspace:
|
|
case Qt::Key_Delete:
|
|
e->ignore(); // don't eat them
|
|
break;
|
|
case Qt::Key_Enter:
|
|
case Qt::Key_Return:
|
|
setActive( i );
|
|
break;
|
|
|
|
default:
|
|
K3ListView::keyPressEvent (e);
|
|
}
|
|
}
|
|
|
|
|
|
#ifdef __GNUC__
|
|
#warning Port this to QDrag once the view doesnot derive from K3ListView any more
|
|
#endif
|
|
Q3DragObject* KNHeaderView::dragObject()
|
|
{
|
|
KNHdrViewItem *item = static_cast<KNHdrViewItem*>( itemAt(viewport()->mapFromGlobal(QCursor::pos())) );
|
|
if (item)
|
|
return item->dragObject();
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
void KNHeaderView::slotSizeChanged( int section, int, int newSize )
|
|
{
|
|
viewport()->repaint( header()->sectionPos(section), 0, newSize, visibleHeight() );
|
|
}
|
|
|
|
|
|
bool KNHeaderView::eventFilter(QObject *o, QEvent *e)
|
|
{
|
|
// right click on header
|
|
if ( e->type() == QEvent::MouseButtonPress &&
|
|
static_cast<QMouseEvent*>(e)->button() == Qt::RightButton &&
|
|
qobject_cast<Q3Header*>( o ) )
|
|
{
|
|
mPopup->popup( static_cast<QMouseEvent*>(e)->globalPos() );
|
|
return true;
|
|
}
|
|
|
|
return K3ListView::eventFilter(o, e);
|
|
}
|
|
|
|
|
|
//BEGIN: KNHeaderViewToolTip ==================================================
|
|
|
|
#ifdef __GNUC__
|
|
#warning Port me!
|
|
#endif
|
|
#if 0
|
|
KNHeaderViewToolTip::KNHeaderViewToolTip( KNHeaderView *parent ) :
|
|
QToolTip( parent->viewport() ),
|
|
listView( parent )
|
|
{
|
|
}
|
|
|
|
|
|
void KNHeaderViewToolTip::maybeTip( const QPoint &p )
|
|
{
|
|
const KNHdrViewItem *item = static_cast<KNHdrViewItem*>( listView->itemAt( p ) );
|
|
if ( !item )
|
|
return;
|
|
const int column = listView->header()->sectionAt( p.x() );
|
|
if ( column == -1 )
|
|
return;
|
|
|
|
if ( !item->showToolTip( column ) )
|
|
return;
|
|
|
|
const QRect itemRect = listView->itemRect( item );
|
|
if ( !itemRect.isValid() )
|
|
return;
|
|
const QRect headerRect = listView->header()->sectionRect( column );
|
|
if ( !headerRect.isValid() )
|
|
return;
|
|
|
|
tip( QRect( headerRect.left(), itemRect.top(), headerRect.width(), itemRect.height() ),
|
|
Qt::escape( item->text( column ) ) );
|
|
}
|
|
#endif
|
|
|
|
//END: KNHeaderViewToolTip ====================================================
|
|
|