/***************************************************************************
* Copyright (C) 2015 by Ilya Kotov *
* forkotov02@hotmail.ru *
* *
* 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. *
* *
* 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, write to the *
* Free Software Foundation, Inc., *
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
***************************************************************************/
#include <QPainter>
#include <QBitmap>
#include <QFont>
#include <QFontMetrics>
#include <QSettings>
#include <QApplication>
#include <QMouseEvent>
#include <QContextMenuEvent>
#include <QMenu>
#include <QLineEdit>
#include <QInputDialog>
#include <QIcon>
#include <QHash>
#include <qmmp/qmmp.h>
#include <qmmpui/playlistmanager.h>
#include <qmmpui/playlistheadermodel.h>
#include <qmmpui/playlistmanager.h>
#include "skin.h"
#include "playlistheader.h"
static const char * const skinned_arrow_down_xpm[] = {
"11 6 2 1",
" c None",
"x c #000000",
" ",
" xxxxxxxxx ",
" xxxxxxx ",
" xxxxx ",
" xxx ",
" x "};
static const char * const skinned_arrow_up_xpm[] = {
"11 6 2 1",
" c None",
"x c #000000",
" x ",
" xxx ",
" xxxxx ",
" xxxxxxx ",
" xxxxxxxxx ",
" "};
PlayListHeader::PlayListHeader(QWidget *parent) :
QWidget(parent)
{
setMouseTracking(true);
m_metrics = 0;
m_padding = 0;
m_pl_padding = 0;
m_number_width = 0;
m_sorting_column = -1;
m_update = false;
m_task = NO_TASK;
m_model = PlayListManager::instance()->headerModel();
m_skin = Skin::instance();
m_menu = new QMenu(this);
m_menu->addAction(QIcon::fromTheme("list-add"), tr("Add Column"), this, SLOT(addColumn()));
m_menu->addAction(QIcon::fromTheme("configure"), tr("Edit Column"), this, SLOT(editColumn()));
m_autoResize = m_menu->addAction(tr("Auto-resize"), this, SLOT(setAutoResize(bool)));
m_autoResize->setCheckable(true);
m_menu->addAction(tr("Restore Size"), this, SLOT(restoreSize()));
m_menu->addSeparator();
m_menu->addAction(QIcon::fromTheme("list-remove"), tr("Remove Column"), this, SLOT(removeColumn()));
connect(m_skin, SIGNAL(skinChanged()), this, SLOT(updateSkin()));
connect(m_model, SIGNAL(columnAdded(int)), SLOT(onColumnAdded(int)));
connect(m_model, SIGNAL(columnRemoved(int)), SLOT(onColumnRemoved(int)));
connect(m_model, SIGNAL(columnMoved(int,int)), SLOT(onColumnMoved(int,int)));
connect(m_model, SIGNAL(columnChanged(int)), SLOT(updateColumns()));
loadColors();
readSettings();
}
PlayListHeader::~PlayListHeader()
{
if (m_metrics)
delete m_metrics;
m_metrics = 0;
writeSettings();
qDeleteAll(m_columns);
m_columns.clear();
}
void PlayListHeader::readSettings()
{
QSettings settings(Qmmp::configFile(), QSettings::IniFormat);
settings.beginGroup("Skinned");
m_font.fromString(settings.value("pl_header_font", qApp->font().toString()).toString());
if (m_metrics)
{
delete m_metrics;
m_metrics = 0;
}
m_metrics = new QFontMetrics(m_font);
m_padding = m_metrics->width("9")/2;
QFont pl_font;
pl_font.fromString(settings.value("pl_font", qApp->font().toString()).toString());
m_pl_padding = QFontMetrics(pl_font).width("9")/2;
if(!m_update)
{
m_model->restoreSettings(&settings);
QList<QVariant> sizes = settings.value("pl_column_sizes").toList();
int autoResizeColumn = settings.value("pl_autoresize_column", -1).toInt();
for(int i = 0; i < m_model->count(); ++i)
{
Column *col = new Column();
if(i < sizes.count())
col->size = sizes.at(i).toInt();
if(i == autoResizeColumn)
col->autoResize = true;
m_columns.append(col);
}
m_update = true;
}
settings.endGroup();
updateColumns();
}
void PlayListHeader::setNumberWidth(int width)
{
if(width != m_number_width)
{
m_number_width = width;
m_columns[0]->minSize = 30 + (m_number_width ? (m_number_width + 2 * m_pl_padding) : 0);
m_columns[0]->size = qMax(m_columns[0]->size, m_columns[0]->minSize);
updateColumns();
}
}
void PlayListHeader::updateColumns()
{
bool rtl = (layoutDirection() == Qt::RightToLeft);
while (m_columns.count() > m_model->count())
delete m_columns.takeFirst();
while(m_columns.count() < m_model->count())
m_columns << new Column();
int sx = 5;
if(m_number_width)
sx += m_number_width + 2 * m_pl_padding;
if(m_model->count() == 1)
{
m_columns[0]->rect = (rtl ? QRect(5, 0, width() - sx - 5, height()) : QRect(sx, 0, width() - sx - 5, height()));
if(m_sorting_column == 0)
{
m_columns[0]->name = m_metrics->elidedText(m_model->name(0), Qt::ElideRight,
m_columns[0]->rect.width() - 2 * m_padding - m_arrow_up.width() - 4);
}
else
{
m_columns[0]->name = m_metrics->elidedText(m_model->name(0), Qt::ElideRight,
m_columns[0]->rect.width() - 2 * m_padding);
}
return;
}
for(int i = 0; i < m_model->count(); ++i)
{
int size = m_columns[i]->size;
//add number width to the first column
if(i == 0 && m_number_width)
size -= m_number_width + 2 * m_pl_padding;
if(rtl)
m_columns[i]->rect = QRect(width() - sx - size, 0, size, height());
else
m_columns[i]->rect = QRect(sx, 0, size, height());
if(i == m_sorting_column)
{
m_columns[i]->name = m_metrics->elidedText(m_model->name(i), Qt::ElideRight,
size - 2 * m_padding - m_arrow_up.width() - 4);
}
else
{
m_columns[i]->name = m_metrics->elidedText(m_model->name(i), Qt::ElideRight,
size - 2 * m_padding);
}
sx += size;
}
update();
}
int PlayListHeader::requiredHeight() const
{
return m_metrics->lineSpacing() + 1;
}
QList<int> PlayListHeader::sizes() const
{
QList<int> sizeList;
for(int i = 0; i < m_columns.size(); ++i)
sizeList.append(m_columns[i]->size);
return sizeList;
}
void PlayListHeader::showSortIndicator(int column, bool reverted)
{
if(m_sorting_column == column && m_reverted == reverted)
return;
m_sorting_column = column;
m_reverted = reverted;
updateColumns();
}
void PlayListHeader::hideSortIndicator()
{
if(m_sorting_column != -1)
{
m_sorting_column = -1;
updateColumns();
}
}
void PlayListHeader::updateSkin()
{
loadColors();
update();
}
void PlayListHeader::addColumn()
{
int column = findColumn(m_pressed_pos);
if(column < 0 && m_pressed_pos.x() > m_columns.last()->rect.right())
column = m_model->count();
else if(column < 0 && m_pressed_pos.x() < m_columns.first()->rect.x())
column = 0;
if(column < 0)
return;
m_model->execInsert(column);
}
void PlayListHeader::editColumn()
{
if(m_pressed_column < 0)
return;
m_model->execEdit(m_pressed_column);
}
void PlayListHeader::removeColumn()
{
if(m_pressed_column < 0)
return;
m_model->remove(m_pressed_column);
}
void PlayListHeader::setAutoResize(bool yes)
{
if(m_pressed_column < 0)
return;
if(yes)
{
for(int i = 0; i < m_columns.count(); ++i)
m_columns[i]->autoResize = false;
}
m_columns[m_pressed_column]->autoResize = yes;
}
void PlayListHeader::restoreSize()
{
if(m_pressed_column < 0)
return;
m_columns[m_pressed_column]->size = 150;
updateColumns();
emit resizeColumnRequest();
}
void PlayListHeader::onColumnAdded(int index)
{
m_columns.insert(index, new Column());
updateColumns();
}
void PlayListHeader::onColumnRemoved(int index)
{
delete m_columns.takeAt(index);
updateColumns();
}
void PlayListHeader::onColumnMoved(int from, int to)
{
//correct geometry
if(from == 0 && m_number_width)
{
m_columns[from]->size -= m_number_width + 2 * m_pl_padding;
m_columns[to]->size += m_number_width + 2 * m_pl_padding;
}
else if(to == 0 && m_number_width)
{
m_columns[from]->size += m_number_width + 2 * m_pl_padding;
m_columns[to]->size -= m_number_width + 2 * m_pl_padding;
}
int minSize = m_columns[from]->minSize;
m_columns[from]->minSize = m_columns[to]->minSize;
m_columns[to]->minSize = minSize;
m_columns.move(from, to);
m_columns[from]->size = qMax(m_columns[from]->size, m_columns[from]->minSize);
m_columns[to]->size = qMax(m_columns[to]->size, m_columns[to]->minSize);
updateColumns();
}
void PlayListHeader::mousePressEvent(QMouseEvent *e)
{
bool rtl = layoutDirection() == Qt::RightToLeft;
if(e->button() == Qt::LeftButton)
{
m_pressed_column = findColumn(e->pos());
if(m_pressed_column >= 0)
{
m_pressed_pos = e->pos();
m_mouse_pos = e->pos();
if(rtl)
{
if(e->pos().x() < m_columns[m_pressed_column]->rect.x() + m_metrics->width("9"))
{
m_old_size = m_columns[m_pressed_column]->size;
m_task = RESIZE;
}
else
{
m_press_offset = e->pos().x() - m_columns[m_pressed_column]->rect.x();
m_task = SORT;
}
}
else
{
if(e->pos().x() > m_columns[m_pressed_column]->rect.right() - m_metrics->width("9"))
{
m_old_size = m_columns[m_pressed_column]->size;
m_task = RESIZE;
}
else
{
m_press_offset = e->pos().x() - m_columns[m_pressed_column]->rect.x();
m_task = SORT;
}
}
}
else
{
m_task = NO_TASK;
update();
}
}
}
void PlayListHeader::mouseReleaseEvent(QMouseEvent *)
{
if(m_task == SORT)
PlayListManager::instance()->selectedPlayList()->sortByColumn(m_pressed_column);
m_task = NO_TASK;
update();
}
void PlayListHeader::mouseMoveEvent(QMouseEvent *e)
{
bool rtl = layoutDirection() == Qt::RightToLeft;
if(m_task == SORT)
{
m_task = MOVE;
}
if(m_task == RESIZE && m_model->count() > 1)
{
if(rtl)
m_columns[m_pressed_column]->size = m_old_size - e->pos().x() + m_pressed_pos.x();
else
m_columns[m_pressed_column]->size = m_old_size + e->pos().x() - m_pressed_pos.x();
m_columns[m_pressed_column]->size = qMax(m_columns[m_pressed_column]->size, m_columns[m_pressed_column]->minSize);
updateColumns();
emit resizeColumnRequest();
}
else if(m_task == MOVE)
{
m_mouse_pos = e->pos();
int dest = -1;
for(int i = 0; i < m_columns.count(); ++i)
{
int x_delta = m_mouse_pos.x() - m_columns[i]->rect.x();
if(x_delta < 0 || x_delta > m_columns[i]->rect.width())
continue;
if(rtl)
{
if((x_delta > m_columns[i]->rect.width()/2 && m_pressed_column > i) ||
(x_delta < m_columns[i]->rect.width()/2 && m_pressed_column < i))
{
dest = i;
break;
}
}
else
{
if((x_delta > m_columns[i]->rect.width()/2 && m_pressed_column < i) ||
(x_delta < m_columns[i]->rect.width()/2 && m_pressed_column > i))
{
dest = i;
break;
}
}
}
if(dest == -1 || dest == m_pressed_column)
{
update();
QWidget::mouseMoveEvent(e);
return;
}
m_model->move(m_pressed_column, dest);
m_pressed_column = dest;
update();
}
else if(e->button() == Qt::NoButton)
{
int column = findColumn(e->pos());
if(rtl)
{
if(column >= 0 && e->pos().x() < m_columns[column]->rect.x() + m_metrics->width("9"))
setCursor(Qt::SplitHCursor);
else
setCursor(Qt::ArrowCursor);
}
else
{
if(column >= 0 && e->pos().x() > m_columns[column]->rect.right() - m_metrics->width("9"))
setCursor(Qt::SplitHCursor);
else
setCursor(Qt::ArrowCursor);
}
}
}
void PlayListHeader::resizeEvent(QResizeEvent *e)
{
if(m_model->count() == 1)
{
updateColumns();
return;
}
int delta = e->size().width() - e->oldSize().width();
int index = -1;
for(int i = 0; i < m_columns.count(); ++i)
{
if(m_columns.at(i)->autoResize)
{
index = i;
break;
}
}
if(index >= 0 && e->oldSize().width() > 10)
{
m_columns[index]->size = qMax(m_columns[index]->minSize, m_columns[index]->size + delta);
updateColumns();
return;
}
if(layoutDirection() == Qt::RightToLeft || e->oldSize().height() != e->size().height())
{
updateColumns();
return;
}
}
void PlayListHeader::contextMenuEvent(QContextMenuEvent *e)
{
m_pressed_pos = e->pos();
m_pressed_column = findColumn(e->pos());
if(m_pressed_column >= 0)
{
m_autoResize->setChecked(m_columns[m_pressed_column]->autoResize);
m_autoResize->setEnabled(true);
foreach (QAction *action, m_menu->actions())
action->setVisible(true);
}
else
{
foreach (QAction *action, m_menu->actions())
{
if(action != m_menu->actions().first())
action->setVisible(false);
}
}
m_menu->exec(e->globalPos());
}
void PlayListHeader::paintEvent(QPaintEvent *)
{
QPainter painter(this);
painter.setBrush(m_normal);
painter.setPen(m_normal);
painter.setFont(m_font);
painter.drawRect(5,-1,width()-10,height()+1);
painter.setPen(m_normal_bg);
bool rtl = (layoutDirection() == Qt::RightToLeft);
if(rtl)
{
if(m_number_width)
{
painter.drawLine(m_columns[0]->rect.right(), 0,
m_columns[0]->rect.right(), height());
}
if(m_columns.count() == 1)
{
painter.drawText(m_columns[0]->rect.right() - m_padding - m_metrics->width(m_columns[0]->name),
m_metrics->ascent(), m_columns[0]->name);
if(m_sorting_column == 0)
{
painter.drawPixmap(m_columns[0]->rect.x() + 4,
(height() - m_arrow_up.height()) / 2,
m_reverted ? m_arrow_up : m_arrow_down);
}
return;
}
for(int i = 0; i < m_columns.count(); ++i)
{
if(m_task == MOVE && i == m_pressed_column)
{
painter.setBrush(m_normal_bg);
painter.setPen(m_current);
painter.drawRect(m_columns[i]->rect.x(), 0,
m_columns[i]->rect.width(), height()-1);
painter.setBrush(m_normal);
painter.setPen(m_normal_bg);
continue;
}
painter.drawText(m_columns[i]->rect.right() - m_padding - m_metrics->width(m_columns[i]->name),
m_metrics->ascent(), m_columns[i]->name);
painter.drawLine(m_columns[i]->rect.x()-1, 0,
m_columns[i]->rect.x()-1, height()+1);
if(i == m_sorting_column)
{
painter.drawPixmap(m_columns[i]->rect.x() + 4,
(height() - m_arrow_up.height()) / 2,
m_reverted ? m_arrow_up : m_arrow_down);
}
}
if(m_task == MOVE)
{
painter.setPen(m_normal);
painter.drawRect(m_mouse_pos.x() - m_press_offset, 0,
m_columns[m_pressed_column]->rect.width(), height());
painter.setPen(m_normal_bg);
painter.drawText(m_mouse_pos.x() - m_press_offset +
m_columns[m_pressed_column]->rect.width() - m_padding -
m_metrics->width(m_columns[m_pressed_column]->name),
m_metrics->ascent(), m_columns[m_pressed_column]->name);
}
}
else
{
if(m_number_width)
{
painter.drawLine(m_columns[0]->rect.x(), 0,
m_columns[0]->rect.x(), height());
}
if(m_columns.count() == 1)
{
painter.drawText(m_columns[0]->rect.x() + m_padding, m_metrics->ascent(), m_columns[0]->name);
if(m_sorting_column == 0)
{
painter.drawPixmap(m_columns[0]->rect.right() - m_arrow_up.width() - 4,
(height() - m_arrow_up.height()) / 2,
m_reverted ? m_arrow_up : m_arrow_down);
}
return;
}
for(int i = 0; i < m_columns.count(); ++i)
{
if(m_task == MOVE && i == m_pressed_column)
{
painter.setBrush(m_normal_bg);
painter.setPen(m_current);
painter.drawRect(m_columns[i]->rect.x(), 0,
m_columns[i]->rect.width(), height()-1);
painter.setBrush(m_normal);
painter.setPen(m_normal_bg);
continue;
}
painter.drawText(m_columns[i]->rect.x() + m_padding, m_metrics->ascent(), m_columns[i]->name);
painter.drawLine(m_columns[i]->rect.right()+1, 0,
m_columns[i]->rect.right()+1, height()+1);
if(i == m_sorting_column)
{
painter.drawPixmap(m_columns[i]->rect.right() - m_arrow_up.width() - 4,
(height() - m_arrow_up.height()) / 2,
m_reverted ? m_arrow_up : m_arrow_down);
}
}
if(m_task == MOVE)
{
painter.setPen(m_normal);
painter.drawRect(m_mouse_pos.x() - m_press_offset, 0,
m_columns[m_pressed_column]->rect.width(), height());
painter.setPen(m_normal_bg);
painter.drawText(m_mouse_pos.x() - m_press_offset + m_padding,
m_metrics->ascent(), m_columns[m_pressed_column]->name);
}
}
}
void PlayListHeader::loadColors()
{
m_normal.setNamedColor(m_skin->getPLValue("normal"));
m_normal_bg.setNamedColor(m_skin->getPLValue("normalbg"));
m_current.setNamedColor(m_skin->getPLValue("current"));
QPixmap px1(skinned_arrow_up_xpm);
QPixmap px2(skinned_arrow_down_xpm);
m_arrow_up = px1;
m_arrow_down = px2;
m_arrow_up.fill(m_normal_bg);
m_arrow_down.fill(m_normal_bg);
m_arrow_up.setMask(px1.createMaskFromColor(Qt::transparent));
m_arrow_down.setMask(px2.createMaskFromColor(Qt::transparent));
}
int PlayListHeader::findColumn(QPoint pos)
{
for(int i = 0; i < m_columns.count(); ++i)
{
if(m_columns[i]->rect.contains(pos))
return i;
}
return -1;
}
void PlayListHeader::writeSettings()
{
QSettings settings(Qmmp::configFile(), QSettings::IniFormat);
settings.beginGroup("Skinned");
m_model->saveSettings(&settings);
QList<QVariant> sizes;
int autoResizeColumn = -1;
for(int i = 0; i < m_columns.count(); ++i)
{
sizes << m_columns[i]->size;
if(m_columns[i]->autoResize)
autoResizeColumn = i;
}
settings.setValue("pl_column_sizes", sizes);
settings.setValue("pl_autoresize_column", autoResizeColumn);
settings.endGroup();
}