mirror of
https://github.com/baldurk/renderdoc.git
synced 2026-05-05 01:20:42 +00:00
540 lines
15 KiB
C++
540 lines
15 KiB
C++
/******************************************************************************
|
|
* The MIT License (MIT)
|
|
*
|
|
* Copyright (c) 2019-2020 Baldur Karlsson
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
******************************************************************************/
|
|
|
|
#include "RDTreeView.h"
|
|
#include <QApplication>
|
|
#include <QClipboard>
|
|
#include <QLabel>
|
|
#include <QMouseEvent>
|
|
#include <QPainter>
|
|
#include <QProxyStyle>
|
|
#include <QStylePainter>
|
|
#include <QToolTip>
|
|
#include <QWheelEvent>
|
|
|
|
static int GetDepth(const QAbstractItemModel *model, const QModelIndex &idx)
|
|
{
|
|
if(idx == QModelIndex())
|
|
return 0;
|
|
|
|
return 1 + GetDepth(model, model->parent(idx));
|
|
}
|
|
|
|
static bool CompareModelIndex(const QModelIndex &a, const QModelIndex &b)
|
|
{
|
|
if(a == b)
|
|
return false;
|
|
|
|
if(a == QModelIndex())
|
|
return true;
|
|
else if(b == QModelIndex())
|
|
return false;
|
|
|
|
if(a.model() != b.model())
|
|
return false;
|
|
|
|
QModelIndex ap = a.model()->parent(a);
|
|
QModelIndex bp = b.model()->parent(b);
|
|
if(ap == bp)
|
|
{
|
|
if(a.row() == b.row())
|
|
return a.column() < b.column();
|
|
return a.row() < b.row();
|
|
}
|
|
|
|
if(a == bp)
|
|
return true;
|
|
if(b == ap)
|
|
return false;
|
|
|
|
int ad = GetDepth(a.model(), a);
|
|
int bd = GetDepth(b.model(), b);
|
|
|
|
if(ad > bd)
|
|
return CompareModelIndex(ap, b);
|
|
else if(ad < bd)
|
|
return CompareModelIndex(a, bp);
|
|
return CompareModelIndex(ap, bp);
|
|
}
|
|
|
|
RDTreeViewDelegate::RDTreeViewDelegate(RDTreeView *view) : ForwardingDelegate(view), m_View(view)
|
|
{
|
|
}
|
|
|
|
QSize RDTreeViewDelegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const
|
|
{
|
|
QSize ret = ForwardingDelegate::sizeHint(option, index);
|
|
|
|
if(m_View->ignoreIconSize())
|
|
ret.setHeight(qMax(option.decorationSize.height() + 2, ret.height()));
|
|
|
|
// expand a pixel for the grid lines
|
|
if(m_View->visibleGridLines())
|
|
ret.setWidth(ret.width() + 1);
|
|
|
|
// ensure we have at least the margin on top of font size. If the style applied more, don't add to
|
|
// it.
|
|
int minHeight = option.fontMetrics.height();
|
|
if(!m_View->ignoreIconSize())
|
|
minHeight = qMax(option.decorationSize.height(), minHeight);
|
|
ret.setHeight(qMax(ret.height(), minHeight + m_View->verticalItemMargin()));
|
|
|
|
return ret;
|
|
}
|
|
|
|
RDTipLabel::RDTipLabel(QWidget *listener) : QLabel(NULL), mouseListener(listener)
|
|
{
|
|
int margin = style()->pixelMetric(QStyle::PM_ToolTipLabelFrameWidth, NULL, this);
|
|
int opacity = style()->styleHint(QStyle::SH_ToolTipLabel_Opacity, NULL, this);
|
|
|
|
setWindowFlags(Qt::ToolTip);
|
|
setAttribute(Qt::WA_TransparentForMouseEvents);
|
|
setForegroundRole(QPalette::ToolTipText);
|
|
setBackgroundRole(QPalette::ToolTipBase);
|
|
setMargin(margin + 1);
|
|
setFrameStyle(QFrame::NoFrame);
|
|
setAlignment(Qt::AlignLeft);
|
|
setIndent(1);
|
|
setWindowOpacity(opacity / 255.0);
|
|
}
|
|
|
|
void RDTipLabel::paintEvent(QPaintEvent *ev)
|
|
{
|
|
QStylePainter p(this);
|
|
QStyleOptionFrame opt;
|
|
opt.init(this);
|
|
p.drawPrimitive(QStyle::PE_PanelTipLabel, opt);
|
|
p.end();
|
|
|
|
QLabel::paintEvent(ev);
|
|
}
|
|
|
|
void RDTipLabel::mousePressEvent(QMouseEvent *e)
|
|
{
|
|
if(mouseListener)
|
|
sendListenerEvent(e);
|
|
}
|
|
|
|
void RDTipLabel::sendListenerEvent(QMouseEvent *e)
|
|
{
|
|
QMouseEvent *duplicate =
|
|
new QMouseEvent(e->type(), mouseListener->mapFromGlobal(e->globalPos()), e->windowPos(),
|
|
e->globalPos(), e->button(), e->buttons(), e->modifiers(), e->source());
|
|
QCoreApplication::postEvent(mouseListener, duplicate);
|
|
}
|
|
|
|
void RDTipLabel::mouseReleaseEvent(QMouseEvent *e)
|
|
{
|
|
if(mouseListener)
|
|
sendListenerEvent(e);
|
|
}
|
|
|
|
void RDTipLabel::mouseDoubleClickEvent(QMouseEvent *e)
|
|
{
|
|
if(mouseListener)
|
|
sendListenerEvent(e);
|
|
}
|
|
|
|
void RDTipLabel::resizeEvent(QResizeEvent *e)
|
|
{
|
|
QStyleHintReturnMask frameMask;
|
|
QStyleOption option;
|
|
option.init(this);
|
|
if(style()->styleHint(QStyle::SH_ToolTip_Mask, &option, this, &frameMask))
|
|
setMask(frameMask.region);
|
|
|
|
QLabel::resizeEvent(e);
|
|
}
|
|
|
|
RDTreeView::RDTreeView(QWidget *parent) : QTreeView(parent)
|
|
{
|
|
setMouseTracking(true);
|
|
|
|
m_delegate = new RDTreeViewDelegate(this);
|
|
QTreeView::setItemDelegate(m_delegate);
|
|
|
|
m_ElidedTooltip = new RDTipLabel(viewport());
|
|
m_ElidedTooltip->hide();
|
|
}
|
|
|
|
RDTreeView::~RDTreeView()
|
|
{
|
|
setModel(NULL);
|
|
|
|
delete m_ElidedTooltip;
|
|
}
|
|
|
|
void RDTreeView::mouseMoveEvent(QMouseEvent *e)
|
|
{
|
|
if(m_ElidedTooltip->isVisible() && !m_ElidedTooltip->geometry().contains(QCursor::pos()))
|
|
m_ElidedTooltip->hide();
|
|
|
|
m_currentHoverIndex = indexAt(e->pos());
|
|
QTreeView::mouseMoveEvent(e);
|
|
}
|
|
|
|
void RDTreeView::wheelEvent(QWheelEvent *e)
|
|
{
|
|
QTreeView::wheelEvent(e);
|
|
m_currentHoverIndex = indexAt(e->pos());
|
|
}
|
|
|
|
void RDTreeView::leaveEvent(QEvent *e)
|
|
{
|
|
if(m_ElidedTooltip->isVisible() && !m_ElidedTooltip->geometry().contains(QCursor::pos()))
|
|
m_ElidedTooltip->hide();
|
|
|
|
m_currentHoverIndex = QModelIndex();
|
|
|
|
emit leave(e);
|
|
|
|
QTreeView::leaveEvent(e);
|
|
}
|
|
|
|
void RDTreeView::keyPressEvent(QKeyEvent *e)
|
|
{
|
|
if(e->matches(QKeySequence::Copy))
|
|
{
|
|
copySelection();
|
|
}
|
|
else
|
|
{
|
|
QTreeView::keyPressEvent(e);
|
|
}
|
|
emit(keyPress(e));
|
|
}
|
|
|
|
bool RDTreeView::viewportEvent(QEvent *event)
|
|
{
|
|
if(m_TooltipElidedItems && event->type() == QEvent::ToolTip)
|
|
{
|
|
QHelpEvent *he = (QHelpEvent *)event;
|
|
QModelIndex index = indexAt(he->pos());
|
|
|
|
QAbstractItemDelegate *delegate = m_userDelegate;
|
|
|
|
if(!delegate)
|
|
delegate = QTreeView::itemDelegate(index);
|
|
|
|
if(delegate)
|
|
{
|
|
QStyleOptionViewItem option;
|
|
option.initFrom(this);
|
|
option.rect = visualRect(index);
|
|
|
|
// delegates get first dibs at processing the event
|
|
bool ret = delegate->helpEvent(he, this, option, index);
|
|
|
|
if(ret)
|
|
return true;
|
|
|
|
QSize desiredSize = delegate->sizeHint(option, index);
|
|
|
|
if(desiredSize.width() > option.rect.width())
|
|
{
|
|
const QString fullText = index.data(Qt::DisplayRole).toString();
|
|
if(!fullText.isEmpty())
|
|
{
|
|
// need to use a custom label tooltip since the QToolTip freaks out as we're placing it
|
|
// underneath the cursor instead of next to it (so that the tooltip lines up over the row)
|
|
m_ElidedTooltip->move(viewport()->mapToGlobal(option.rect.topLeft()));
|
|
m_ElidedTooltip->setText(fullText);
|
|
m_ElidedTooltip->show();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return QTreeView::viewportEvent(event);
|
|
}
|
|
|
|
void RDTreeView::setItemDelegate(QAbstractItemDelegate *delegate)
|
|
{
|
|
m_userDelegate = delegate;
|
|
m_delegate->setForwardDelegate(m_userDelegate);
|
|
}
|
|
|
|
QAbstractItemDelegate *RDTreeView::itemDelegate() const
|
|
{
|
|
return m_userDelegate;
|
|
}
|
|
|
|
void RDTreeView::setModel(QAbstractItemModel *model)
|
|
{
|
|
QAbstractItemModel *old = this->model();
|
|
|
|
if(old)
|
|
QObject::disconnect(old, &QAbstractItemModel::modelAboutToBeReset, this,
|
|
&RDTreeView::modelAboutToBeReset);
|
|
|
|
QTreeView::setModel(model);
|
|
|
|
if(model)
|
|
{
|
|
QObject::connect(model, &QAbstractItemModel::modelAboutToBeReset, this,
|
|
&RDTreeView::modelAboutToBeReset);
|
|
}
|
|
}
|
|
|
|
void RDTreeView::modelAboutToBeReset()
|
|
{
|
|
m_currentHoverIndex = QModelIndex();
|
|
}
|
|
|
|
void RDTreeView::saveExpansion(RDTreeViewExpansionState &state, const ExpansionKeyGen &keygen)
|
|
{
|
|
state.clear();
|
|
|
|
for(int i = 0; i < model()->rowCount(); i++)
|
|
saveExpansionFromRow(state, model()->index(i, 0), 0, keygen);
|
|
}
|
|
|
|
void RDTreeView::applyExpansion(const RDTreeViewExpansionState &state, const ExpansionKeyGen &keygen)
|
|
{
|
|
for(int i = 0; i < model()->rowCount(); i++)
|
|
applyExpansionToRow(state, model()->index(i, 0), 0, keygen);
|
|
}
|
|
|
|
void RDTreeView::copySelection()
|
|
{
|
|
QModelIndexList sel = selectionModel()->selectedRows();
|
|
|
|
std::sort(sel.begin(), sel.end(), CompareModelIndex);
|
|
|
|
QVector<int> widths;
|
|
|
|
ICaptureContext *ctx = getCaptureContext(this);
|
|
|
|
int minDepth = INT_MAX;
|
|
int maxDepth = 0;
|
|
|
|
// align the copied data so that each column is the same width
|
|
for(QModelIndex idx : sel)
|
|
{
|
|
int colCount = model()->columnCount(idx);
|
|
|
|
widths.resize(qMax(widths.size(), colCount));
|
|
|
|
for(int i = 0; i < colCount; i++)
|
|
{
|
|
QVariant var = model()->data(model()->index(idx.row(), i, idx.parent()));
|
|
QString text = ctx ? RichResourceTextFormat(*ctx, var) : var.toString();
|
|
widths[i] = qMax(widths[i], text.count());
|
|
}
|
|
|
|
int depth = GetDepth(model(), idx);
|
|
|
|
minDepth = qMin(minDepth, depth);
|
|
maxDepth = qMax(maxDepth, depth);
|
|
}
|
|
|
|
// add on two characters for every depth, for indent
|
|
for(int &i : widths)
|
|
i += 2 * (maxDepth - minDepth - 1);
|
|
|
|
// only align up to 50 characters so one really long item doesn't mess up the whole thing
|
|
for(int &i : widths)
|
|
i = qMin(50, i);
|
|
|
|
QString clipData;
|
|
for(QModelIndex idx : sel)
|
|
{
|
|
int colCount = model()->columnCount(idx);
|
|
|
|
int depth = GetDepth(model(), idx);
|
|
|
|
for(int i = 0; i < colCount; i++)
|
|
{
|
|
QString format = i == 0 ? QFormatStr("%1") : QFormatStr(" %1");
|
|
|
|
QVariant var = model()->data(model()->index(idx.row(), i, idx.parent()));
|
|
QString text = ctx ? RichResourceTextFormat(*ctx, var) : var.toString();
|
|
|
|
if(i == 0)
|
|
text.prepend(QString((depth - minDepth) * 2, QLatin1Char(' ')));
|
|
|
|
clipData += format.arg(text, -widths[i]);
|
|
}
|
|
|
|
clipData += lit("\n");
|
|
}
|
|
|
|
QClipboard *clipboard = QApplication::clipboard();
|
|
clipboard->setText(clipData.trimmed());
|
|
}
|
|
|
|
void RDTreeView::saveExpansionFromRow(RDTreeViewExpansionState &state, QModelIndex idx, uint seed,
|
|
const ExpansionKeyGen &keygen)
|
|
{
|
|
if(!idx.isValid())
|
|
return;
|
|
|
|
uint key = keygen(idx, seed);
|
|
if(isExpanded(idx))
|
|
{
|
|
state.insert(key);
|
|
|
|
// only recurse to children if this one is expanded - forget expansion state under collapsed
|
|
// branches. Technically we're losing information here but it allows us to skip a full expensive
|
|
// search
|
|
for(int i = 0; i < model()->rowCount(idx); i++)
|
|
saveExpansionFromRow(state, model()->index(i, 0, idx), seed, keygen);
|
|
}
|
|
}
|
|
|
|
void RDTreeView::applyExpansionToRow(const RDTreeViewExpansionState &state, QModelIndex idx,
|
|
uint seed, const ExpansionKeyGen &keygen)
|
|
{
|
|
if(!idx.isValid())
|
|
return;
|
|
|
|
uint key = keygen(idx, seed);
|
|
if(state.contains(key))
|
|
{
|
|
expand(idx);
|
|
|
|
// same as above - only recurse when we have a parent that's expanded.
|
|
for(int i = 0; i < model()->rowCount(idx); i++)
|
|
applyExpansionToRow(state, model()->index(i, 0, idx), seed, keygen);
|
|
}
|
|
}
|
|
|
|
void RDTreeView::drawRow(QPainter *painter, const QStyleOptionViewItem &options,
|
|
const QModelIndex &index) const
|
|
{
|
|
QTreeView::drawRow(painter, options, index);
|
|
|
|
if(m_VisibleGridLines)
|
|
{
|
|
QPen p = painter->pen();
|
|
|
|
QColor back = options.palette.color(QPalette::Active, QPalette::Background);
|
|
QColor fore = options.palette.color(QPalette::Active, QPalette::Foreground);
|
|
|
|
// draw the grid lines with a colour half way between background and foreground
|
|
painter->setPen(QPen(QColor::fromRgbF(back.redF() * 0.8 + fore.redF() * 0.2,
|
|
back.greenF() * 0.8 + fore.greenF() * 0.2,
|
|
back.blueF() * 0.8 + fore.blueF() * 0.2)));
|
|
|
|
QRect intersectrect = options.rect.adjusted(0, 0, 1, 0);
|
|
|
|
for(int i = 0, count = model()->columnCount(); i < count; i++)
|
|
{
|
|
QRect r = visualRect(model()->index(index.row(), i, index.parent()));
|
|
|
|
if(r.width() <= 0)
|
|
r.moveLeft(r.left() + r.width());
|
|
if(r.height() <= 0)
|
|
r.moveTop(r.top() + r.height());
|
|
|
|
r = r.intersected(intersectrect);
|
|
|
|
if(treePosition() == i)
|
|
{
|
|
int depth = 1;
|
|
QModelIndex idx = index;
|
|
while(idx.parent().isValid())
|
|
{
|
|
depth++;
|
|
idx = idx.parent();
|
|
}
|
|
r.setLeft(r.left() - indentation() * depth);
|
|
}
|
|
|
|
// draw bottom and right of the rect
|
|
painter->drawLine(r.bottomLeft(), r.bottomRight());
|
|
painter->drawLine(r.topRight(), r.bottomRight());
|
|
}
|
|
|
|
painter->setPen(p);
|
|
}
|
|
}
|
|
|
|
void RDTreeView::fillBranchesRect(QPainter *painter, const QRect &rect, const QModelIndex &index) const
|
|
{
|
|
QStyleOptionViewItem opt;
|
|
opt.initFrom(this);
|
|
if(selectionModel()->isSelected(index))
|
|
opt.state |= QStyle::State_Selected;
|
|
if(m_currentHoverIndex.row() == index.row() && m_currentHoverIndex.parent() == index.parent())
|
|
opt.state |= QStyle::State_MouseOver;
|
|
else
|
|
opt.state &= ~QStyle::State_MouseOver;
|
|
|
|
if(hasFocus())
|
|
opt.state |= (QStyle::State_Active | QStyle::State_HasFocus);
|
|
else
|
|
opt.state &= ~(QStyle::State_Active | QStyle::State_HasFocus);
|
|
|
|
int depth = 1;
|
|
QModelIndex idx = index.parent();
|
|
while(idx.isValid())
|
|
{
|
|
depth++;
|
|
idx = idx.parent();
|
|
}
|
|
|
|
opt.rect = rect;
|
|
opt.rect.setWidth(depth * indentation());
|
|
opt.showDecorationSelected = true;
|
|
style()->drawPrimitive(QStyle::PE_PanelItemViewItem, &opt, painter, this);
|
|
}
|
|
|
|
void RDTreeView::drawBranches(QPainter *painter, const QRect &rect, const QModelIndex &index) const
|
|
{
|
|
if(m_fillBranchRect)
|
|
fillBranchesRect(painter, rect, index);
|
|
|
|
if(m_VisibleBranches)
|
|
{
|
|
QTreeView::drawBranches(painter, rect, index);
|
|
}
|
|
else
|
|
{
|
|
// draw only the expand item, not the branches
|
|
QRect primitive(0, rect.top(), qMin(rect.width(), indentation()), rect.height());
|
|
|
|
// if root isn't decorated, skip
|
|
if(!rootIsDecorated() && !index.parent().isValid())
|
|
return;
|
|
|
|
// if no children, nothing to render
|
|
if(model()->rowCount(index) == 0)
|
|
return;
|
|
|
|
QStyleOptionViewItem opt = viewOptions();
|
|
|
|
opt.rect = primitive;
|
|
|
|
// unfortunately QStyle::State_Children doesn't render ONLY the
|
|
// open-toggle-button, but the vertical line upwards to a previous sibling.
|
|
// For consistency, draw one downwards too.
|
|
opt.state = QStyle::State_Children | QStyle::State_Sibling;
|
|
if(isExpanded(index))
|
|
opt.state |= QStyle::State_Open;
|
|
|
|
style()->drawPrimitive(QStyle::PE_IndicatorBranch, &opt, painter, this);
|
|
}
|
|
}
|