mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-25 10:43:19 -03:00
Merge bitcoin-core/gui#18: Add peertablesortproxy module
5a4a15d2b4
qt, refactor: Drop no longer used PeerTableModel::getRowByNodeId func (Hennadii Stepanov)9a9f180df0
qt, refactor: Drop no longer used PeerTableModel::sort function (Hennadii Stepanov)778a64af20
qt: Use PeerTableSortProxy for sorting peer table (Hennadii Stepanov)df2d165ba9
qt: Add peertablesortproxy module (Hennadii Stepanov) Pull request description: The "Peers" table in the "Node" window does not hold multiple selection after sorting. This PR introduces a `QSortFilterProxyModel` subclass, that is a standard Qt [practice](https://doc.qt.io/qt-5/model-view-programming.html#custom-sorting-models) for such cases. Now the sorting code is encapsulated into the dedicated Qt class, and we do not need to maintain it. Fixes #283 (additionally). --- On **master** (7ae86b3c68
): - rows are sorted by "Ping", and a selection is made ![Screenshot from 2020-11-28 22-53-11](https://user-images.githubusercontent.com/32963518/100525900-96eaed00-31cc-11eb-86e7-72ede3b8b33c.png) - rows are sorted by "NodeId", and the previous selection is _lost_ ![Screenshot from 2020-11-28 22-53-21](https://user-images.githubusercontent.com/32963518/100525904-9c483780-31cc-11eb-957c-06f53d7d31ab.png) With **this PR**: - rows are sorted by "Ping", and a selection is made ![Screenshot from 2020-11-28 22-39-41](https://user-images.githubusercontent.com/32963518/100525776-06aca800-31cc-11eb-8c4e-9c6566fe80fe.png) - rows are sorted by "NodeId", and the row are still selected ![Screenshot from 2020-11-28 22-39-53](https://user-images.githubusercontent.com/32963518/100525791-2348e000-31cc-11eb-8b78-716a5551d7ec.png) ACKs for top commit: jarolrod: re-ACK5a4a15d2b4
, tested on macOS 11.2 Qt 5.15.2 after rebase promag: Tested ACK5a4a15d2b4
. Tree-SHA512: f81c1385892fbf1a46ffb98b42094ca1cc97da52114bbbc94fedb553899b1f18c26a349e186bba6e27922a89426bd61e8bc88b1f7832512dbe211b5f834e076e
This commit is contained in:
commit
328da33557
10 changed files with 94 additions and 156 deletions
|
@ -34,6 +34,7 @@
|
||||||
<ClCompile Include="..\..\src\qt\overviewpage.cpp" />
|
<ClCompile Include="..\..\src\qt\overviewpage.cpp" />
|
||||||
<ClCompile Include="..\..\src\qt\paymentserver.cpp" />
|
<ClCompile Include="..\..\src\qt\paymentserver.cpp" />
|
||||||
<ClCompile Include="..\..\src\qt\peertablemodel.cpp" />
|
<ClCompile Include="..\..\src\qt\peertablemodel.cpp" />
|
||||||
|
<ClCompile Include="..\..\src\qt\peertablesortproxy.cpp" />
|
||||||
<ClCompile Include="..\..\src\qt\platformstyle.cpp" />
|
<ClCompile Include="..\..\src\qt\platformstyle.cpp" />
|
||||||
<ClCompile Include="..\..\src\qt\psbtoperationsdialog.cpp" />
|
<ClCompile Include="..\..\src\qt\psbtoperationsdialog.cpp" />
|
||||||
<ClCompile Include="..\..\src\qt\qrimagewidget.cpp" />
|
<ClCompile Include="..\..\src\qt\qrimagewidget.cpp" />
|
||||||
|
@ -87,6 +88,7 @@
|
||||||
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_overviewpage.cpp" />
|
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_overviewpage.cpp" />
|
||||||
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_paymentserver.cpp" />
|
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_paymentserver.cpp" />
|
||||||
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_peertablemodel.cpp" />
|
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_peertablemodel.cpp" />
|
||||||
|
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_peertablesortproxy.cpp" />
|
||||||
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_platformstyle.cpp" />
|
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_platformstyle.cpp" />
|
||||||
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_psbtoperationsdialog.cpp" />
|
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_psbtoperationsdialog.cpp" />
|
||||||
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_qrimagewidget.cpp" />
|
<ClCompile Include="$(GeneratedFilesOutDir)\moc\moc_qrimagewidget.cpp" />
|
||||||
|
|
|
@ -61,6 +61,7 @@ QT_MOC_CPP = \
|
||||||
qt/moc_optionsmodel.cpp \
|
qt/moc_optionsmodel.cpp \
|
||||||
qt/moc_overviewpage.cpp \
|
qt/moc_overviewpage.cpp \
|
||||||
qt/moc_peertablemodel.cpp \
|
qt/moc_peertablemodel.cpp \
|
||||||
|
qt/moc_peertablesortproxy.cpp \
|
||||||
qt/moc_paymentserver.cpp \
|
qt/moc_paymentserver.cpp \
|
||||||
qt/moc_psbtoperationsdialog.cpp \
|
qt/moc_psbtoperationsdialog.cpp \
|
||||||
qt/moc_qrimagewidget.cpp \
|
qt/moc_qrimagewidget.cpp \
|
||||||
|
@ -134,6 +135,7 @@ BITCOIN_QT_H = \
|
||||||
qt/overviewpage.h \
|
qt/overviewpage.h \
|
||||||
qt/paymentserver.h \
|
qt/paymentserver.h \
|
||||||
qt/peertablemodel.h \
|
qt/peertablemodel.h \
|
||||||
|
qt/peertablesortproxy.h \
|
||||||
qt/platformstyle.h \
|
qt/platformstyle.h \
|
||||||
qt/psbtoperationsdialog.h \
|
qt/psbtoperationsdialog.h \
|
||||||
qt/qrimagewidget.h \
|
qt/qrimagewidget.h \
|
||||||
|
@ -232,6 +234,7 @@ BITCOIN_QT_BASE_CPP = \
|
||||||
qt/optionsdialog.cpp \
|
qt/optionsdialog.cpp \
|
||||||
qt/optionsmodel.cpp \
|
qt/optionsmodel.cpp \
|
||||||
qt/peertablemodel.cpp \
|
qt/peertablemodel.cpp \
|
||||||
|
qt/peertablesortproxy.cpp \
|
||||||
qt/platformstyle.cpp \
|
qt/platformstyle.cpp \
|
||||||
qt/qvalidatedlineedit.cpp \
|
qt/qvalidatedlineedit.cpp \
|
||||||
qt/qvaluecombobox.cpp \
|
qt/qvaluecombobox.cpp \
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
#include <qt/guiconstants.h>
|
#include <qt/guiconstants.h>
|
||||||
#include <qt/guiutil.h>
|
#include <qt/guiutil.h>
|
||||||
#include <qt/peertablemodel.h>
|
#include <qt/peertablemodel.h>
|
||||||
|
#include <qt/peertablesortproxy.h>
|
||||||
|
|
||||||
#include <clientversion.h>
|
#include <clientversion.h>
|
||||||
#include <interfaces/handler.h>
|
#include <interfaces/handler.h>
|
||||||
|
@ -38,7 +39,11 @@ ClientModel::ClientModel(interfaces::Node& node, OptionsModel *_optionsModel, QO
|
||||||
{
|
{
|
||||||
cachedBestHeaderHeight = -1;
|
cachedBestHeaderHeight = -1;
|
||||||
cachedBestHeaderTime = -1;
|
cachedBestHeaderTime = -1;
|
||||||
|
|
||||||
peerTableModel = new PeerTableModel(m_node, this);
|
peerTableModel = new PeerTableModel(m_node, this);
|
||||||
|
m_peer_table_sort_proxy = new PeerTableSortProxy(this);
|
||||||
|
m_peer_table_sort_proxy->setSourceModel(peerTableModel);
|
||||||
|
|
||||||
banTableModel = new BanTableModel(m_node, this);
|
banTableModel = new BanTableModel(m_node, this);
|
||||||
|
|
||||||
QTimer* timer = new QTimer;
|
QTimer* timer = new QTimer;
|
||||||
|
@ -184,6 +189,11 @@ PeerTableModel *ClientModel::getPeerTableModel()
|
||||||
return peerTableModel;
|
return peerTableModel;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PeerTableSortProxy* ClientModel::peerTableSortProxy()
|
||||||
|
{
|
||||||
|
return m_peer_table_sort_proxy;
|
||||||
|
}
|
||||||
|
|
||||||
BanTableModel *ClientModel::getBanTableModel()
|
BanTableModel *ClientModel::getBanTableModel()
|
||||||
{
|
{
|
||||||
return banTableModel;
|
return banTableModel;
|
||||||
|
|
|
@ -17,6 +17,7 @@ class BanTableModel;
|
||||||
class CBlockIndex;
|
class CBlockIndex;
|
||||||
class OptionsModel;
|
class OptionsModel;
|
||||||
class PeerTableModel;
|
class PeerTableModel;
|
||||||
|
class PeerTableSortProxy;
|
||||||
enum class SynchronizationState;
|
enum class SynchronizationState;
|
||||||
|
|
||||||
namespace interfaces {
|
namespace interfaces {
|
||||||
|
@ -54,6 +55,7 @@ public:
|
||||||
interfaces::Node& node() const { return m_node; }
|
interfaces::Node& node() const { return m_node; }
|
||||||
OptionsModel *getOptionsModel();
|
OptionsModel *getOptionsModel();
|
||||||
PeerTableModel *getPeerTableModel();
|
PeerTableModel *getPeerTableModel();
|
||||||
|
PeerTableSortProxy* peerTableSortProxy();
|
||||||
BanTableModel *getBanTableModel();
|
BanTableModel *getBanTableModel();
|
||||||
|
|
||||||
//! Return number of connections, default is in- and outbound (total)
|
//! Return number of connections, default is in- and outbound (total)
|
||||||
|
@ -96,6 +98,7 @@ private:
|
||||||
std::unique_ptr<interfaces::Handler> m_handler_notify_header_tip;
|
std::unique_ptr<interfaces::Handler> m_handler_notify_header_tip;
|
||||||
OptionsModel *optionsModel;
|
OptionsModel *optionsModel;
|
||||||
PeerTableModel *peerTableModel;
|
PeerTableModel *peerTableModel;
|
||||||
|
PeerTableSortProxy* m_peer_table_sort_proxy{nullptr};
|
||||||
BanTableModel *banTableModel;
|
BanTableModel *banTableModel;
|
||||||
|
|
||||||
//! A thread to interact with m_node asynchronously
|
//! A thread to interact with m_node asynchronously
|
||||||
|
|
|
@ -11,55 +11,18 @@
|
||||||
|
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
|
||||||
#include <QDebug>
|
|
||||||
#include <QList>
|
#include <QList>
|
||||||
#include <QTimer>
|
#include <QTimer>
|
||||||
|
|
||||||
bool NodeLessThan::operator()(const CNodeCombinedStats &left, const CNodeCombinedStats &right) const
|
|
||||||
{
|
|
||||||
const CNodeStats *pLeft = &(left.nodeStats);
|
|
||||||
const CNodeStats *pRight = &(right.nodeStats);
|
|
||||||
|
|
||||||
if (order == Qt::DescendingOrder)
|
|
||||||
std::swap(pLeft, pRight);
|
|
||||||
|
|
||||||
switch (static_cast<PeerTableModel::ColumnIndex>(column)) {
|
|
||||||
case PeerTableModel::NetNodeId:
|
|
||||||
return pLeft->nodeid < pRight->nodeid;
|
|
||||||
case PeerTableModel::Address:
|
|
||||||
return pLeft->addrName.compare(pRight->addrName) < 0;
|
|
||||||
case PeerTableModel::ConnectionType:
|
|
||||||
return pLeft->m_conn_type < pRight->m_conn_type;
|
|
||||||
case PeerTableModel::Network:
|
|
||||||
return pLeft->m_network < pRight->m_network;
|
|
||||||
case PeerTableModel::Ping:
|
|
||||||
return pLeft->m_min_ping_time < pRight->m_min_ping_time;
|
|
||||||
case PeerTableModel::Sent:
|
|
||||||
return pLeft->nSendBytes < pRight->nSendBytes;
|
|
||||||
case PeerTableModel::Received:
|
|
||||||
return pLeft->nRecvBytes < pRight->nRecvBytes;
|
|
||||||
case PeerTableModel::Subversion:
|
|
||||||
return pLeft->cleanSubVer.compare(pRight->cleanSubVer) < 0;
|
|
||||||
} // no default case, so the compiler can warn about missing cases
|
|
||||||
assert(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
// private implementation
|
// private implementation
|
||||||
class PeerTablePriv
|
class PeerTablePriv
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
/** Local cache of peer information */
|
/** Local cache of peer information */
|
||||||
QList<CNodeCombinedStats> cachedNodeStats;
|
QList<CNodeCombinedStats> cachedNodeStats;
|
||||||
/** Column to sort nodes by (default to unsorted) */
|
|
||||||
int sortColumn{-1};
|
|
||||||
/** Order (ascending or descending) to sort nodes by */
|
|
||||||
Qt::SortOrder sortOrder;
|
|
||||||
/** Index of rows by node ID */
|
|
||||||
std::map<NodeId, int> mapNodeRows;
|
|
||||||
|
|
||||||
/** Pull a full list of peers from vNodes into our cache */
|
/** Pull a full list of peers from vNodes into our cache */
|
||||||
void refreshPeers(interfaces::Node& node)
|
void refreshPeers(interfaces::Node& node)
|
||||||
{
|
|
||||||
{
|
{
|
||||||
cachedNodeStats.clear();
|
cachedNodeStats.clear();
|
||||||
|
|
||||||
|
@ -76,17 +39,6 @@ public:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sortColumn >= 0)
|
|
||||||
// sort cacheNodeStats (use stable sort to prevent rows jumping around unnecessarily)
|
|
||||||
std::stable_sort(cachedNodeStats.begin(), cachedNodeStats.end(), NodeLessThan(sortColumn, sortOrder));
|
|
||||||
|
|
||||||
// build index map
|
|
||||||
mapNodeRows.clear();
|
|
||||||
int row = 0;
|
|
||||||
for (const CNodeCombinedStats& stats : cachedNodeStats)
|
|
||||||
mapNodeRows.insert(std::pair<NodeId, int>(stats.nodeStats.nodeid, row++));
|
|
||||||
}
|
|
||||||
|
|
||||||
int size() const
|
int size() const
|
||||||
{
|
{
|
||||||
return cachedNodeStats.size();
|
return cachedNodeStats.size();
|
||||||
|
@ -194,10 +146,7 @@ QVariant PeerTableModel::data(const QModelIndex &index, int role) const
|
||||||
} // no default case, so the compiler can warn about missing cases
|
} // no default case, so the compiler can warn about missing cases
|
||||||
assert(false);
|
assert(false);
|
||||||
} else if (role == StatsRole) {
|
} else if (role == StatsRole) {
|
||||||
switch (index.column()) {
|
return QVariant::fromValue(rec);
|
||||||
case NetNodeId: return QVariant::fromValue(rec);
|
|
||||||
default: return QVariant();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return QVariant();
|
return QVariant();
|
||||||
|
@ -239,19 +188,3 @@ void PeerTableModel::refresh()
|
||||||
priv->refreshPeers(m_node);
|
priv->refreshPeers(m_node);
|
||||||
Q_EMIT layoutChanged();
|
Q_EMIT layoutChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
int PeerTableModel::getRowByNodeId(NodeId nodeid)
|
|
||||||
{
|
|
||||||
std::map<NodeId, int>::iterator it = priv->mapNodeRows.find(nodeid);
|
|
||||||
if (it == priv->mapNodeRows.end())
|
|
||||||
return -1;
|
|
||||||
|
|
||||||
return it->second;
|
|
||||||
}
|
|
||||||
|
|
||||||
void PeerTableModel::sort(int column, Qt::SortOrder order)
|
|
||||||
{
|
|
||||||
priv->sortColumn = column;
|
|
||||||
priv->sortOrder = order;
|
|
||||||
refresh();
|
|
||||||
}
|
|
||||||
|
|
|
@ -30,18 +30,6 @@ struct CNodeCombinedStats {
|
||||||
};
|
};
|
||||||
Q_DECLARE_METATYPE(CNodeCombinedStats*)
|
Q_DECLARE_METATYPE(CNodeCombinedStats*)
|
||||||
|
|
||||||
class NodeLessThan
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
NodeLessThan(int nColumn, Qt::SortOrder fOrder) :
|
|
||||||
column(nColumn), order(fOrder) {}
|
|
||||||
bool operator()(const CNodeCombinedStats &left, const CNodeCombinedStats &right) const;
|
|
||||||
|
|
||||||
private:
|
|
||||||
int column;
|
|
||||||
Qt::SortOrder order;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Qt model providing information about connected peers, similar to the
|
Qt model providing information about connected peers, similar to the
|
||||||
"getpeerinfo" RPC call. Used by the rpc console UI.
|
"getpeerinfo" RPC call. Used by the rpc console UI.
|
||||||
|
@ -53,7 +41,6 @@ class PeerTableModel : public QAbstractTableModel
|
||||||
public:
|
public:
|
||||||
explicit PeerTableModel(interfaces::Node& node, QObject* parent);
|
explicit PeerTableModel(interfaces::Node& node, QObject* parent);
|
||||||
~PeerTableModel();
|
~PeerTableModel();
|
||||||
int getRowByNodeId(NodeId nodeid);
|
|
||||||
void startAutoRefresh();
|
void startAutoRefresh();
|
||||||
void stopAutoRefresh();
|
void stopAutoRefresh();
|
||||||
|
|
||||||
|
@ -80,7 +67,6 @@ public:
|
||||||
QVariant headerData(int section, Qt::Orientation orientation, int role) const override;
|
QVariant headerData(int section, Qt::Orientation orientation, int role) const override;
|
||||||
QModelIndex index(int row, int column, const QModelIndex &parent) const override;
|
QModelIndex index(int row, int column, const QModelIndex &parent) const override;
|
||||||
Qt::ItemFlags flags(const QModelIndex &index) const override;
|
Qt::ItemFlags flags(const QModelIndex &index) const override;
|
||||||
void sort(int column, Qt::SortOrder order) override;
|
|
||||||
/*@}*/
|
/*@}*/
|
||||||
|
|
||||||
public Q_SLOTS:
|
public Q_SLOTS:
|
||||||
|
|
43
src/qt/peertablesortproxy.cpp
Normal file
43
src/qt/peertablesortproxy.cpp
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
// Copyright (c) 2020 The Bitcoin Core developers
|
||||||
|
// Distributed under the MIT software license, see the accompanying
|
||||||
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
||||||
|
|
||||||
|
#include <qt/peertablesortproxy.h>
|
||||||
|
|
||||||
|
#include <qt/peertablemodel.h>
|
||||||
|
#include <util/check.h>
|
||||||
|
|
||||||
|
#include <QModelIndex>
|
||||||
|
#include <QString>
|
||||||
|
#include <QVariant>
|
||||||
|
|
||||||
|
PeerTableSortProxy::PeerTableSortProxy(QObject* parent)
|
||||||
|
: QSortFilterProxyModel(parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PeerTableSortProxy::lessThan(const QModelIndex& left_index, const QModelIndex& right_index) const
|
||||||
|
{
|
||||||
|
const CNodeStats left_stats = Assert(sourceModel()->data(left_index, PeerTableModel::StatsRole).value<CNodeCombinedStats*>())->nodeStats;
|
||||||
|
const CNodeStats right_stats = Assert(sourceModel()->data(right_index, PeerTableModel::StatsRole).value<CNodeCombinedStats*>())->nodeStats;
|
||||||
|
|
||||||
|
switch (static_cast<PeerTableModel::ColumnIndex>(left_index.column())) {
|
||||||
|
case PeerTableModel::NetNodeId:
|
||||||
|
return left_stats.nodeid < right_stats.nodeid;
|
||||||
|
case PeerTableModel::Address:
|
||||||
|
return left_stats.addrName.compare(right_stats.addrName) < 0;
|
||||||
|
case PeerTableModel::ConnectionType:
|
||||||
|
return left_stats.m_conn_type < right_stats.m_conn_type;
|
||||||
|
case PeerTableModel::Network:
|
||||||
|
return left_stats.m_network < right_stats.m_network;
|
||||||
|
case PeerTableModel::Ping:
|
||||||
|
return left_stats.m_min_ping_time < right_stats.m_min_ping_time;
|
||||||
|
case PeerTableModel::Sent:
|
||||||
|
return left_stats.nSendBytes < right_stats.nSendBytes;
|
||||||
|
case PeerTableModel::Received:
|
||||||
|
return left_stats.nRecvBytes < right_stats.nRecvBytes;
|
||||||
|
case PeerTableModel::Subversion:
|
||||||
|
return left_stats.cleanSubVer.compare(right_stats.cleanSubVer) < 0;
|
||||||
|
} // no default case, so the compiler can warn about missing cases
|
||||||
|
assert(false);
|
||||||
|
}
|
25
src/qt/peertablesortproxy.h
Normal file
25
src/qt/peertablesortproxy.h
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
// Copyright (c) 2020 The Bitcoin Core developers
|
||||||
|
// Distributed under the MIT software license, see the accompanying
|
||||||
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
||||||
|
|
||||||
|
#ifndef BITCOIN_QT_PEERTABLESORTPROXY_H
|
||||||
|
#define BITCOIN_QT_PEERTABLESORTPROXY_H
|
||||||
|
|
||||||
|
#include <QSortFilterProxyModel>
|
||||||
|
|
||||||
|
QT_BEGIN_NAMESPACE
|
||||||
|
class QModelIndex;
|
||||||
|
QT_END_NAMESPACE
|
||||||
|
|
||||||
|
class PeerTableSortProxy : public QSortFilterProxyModel
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit PeerTableSortProxy(QObject* parent = nullptr);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
bool lessThan(const QModelIndex& left_index, const QModelIndex& right_index) const override;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // BITCOIN_QT_PEERTABLESORTPROXY_H
|
|
@ -9,13 +9,14 @@
|
||||||
#include <qt/rpcconsole.h>
|
#include <qt/rpcconsole.h>
|
||||||
#include <qt/forms/ui_debugwindow.h>
|
#include <qt/forms/ui_debugwindow.h>
|
||||||
|
|
||||||
#include <qt/bantablemodel.h>
|
|
||||||
#include <qt/clientmodel.h>
|
|
||||||
#include <qt/platformstyle.h>
|
|
||||||
#include <qt/walletmodel.h>
|
|
||||||
#include <chainparams.h>
|
#include <chainparams.h>
|
||||||
#include <interfaces/node.h>
|
#include <interfaces/node.h>
|
||||||
#include <netbase.h>
|
#include <netbase.h>
|
||||||
|
#include <qt/bantablemodel.h>
|
||||||
|
#include <qt/clientmodel.h>
|
||||||
|
#include <qt/peertablesortproxy.h>
|
||||||
|
#include <qt/platformstyle.h>
|
||||||
|
#include <qt/walletmodel.h>
|
||||||
#include <rpc/client.h>
|
#include <rpc/client.h>
|
||||||
#include <rpc/server.h>
|
#include <rpc/server.h>
|
||||||
#include <util/strencodings.h>
|
#include <util/strencodings.h>
|
||||||
|
@ -606,7 +607,7 @@ void RPCConsole::setClientModel(ClientModel *model, int bestblock_height, int64_
|
||||||
connect(model, &ClientModel::mempoolSizeChanged, this, &RPCConsole::setMempoolSize);
|
connect(model, &ClientModel::mempoolSizeChanged, this, &RPCConsole::setMempoolSize);
|
||||||
|
|
||||||
// set up peer table
|
// set up peer table
|
||||||
ui->peerWidget->setModel(model->getPeerTableModel());
|
ui->peerWidget->setModel(model->peerTableSortProxy());
|
||||||
ui->peerWidget->verticalHeader()->hide();
|
ui->peerWidget->verticalHeader()->hide();
|
||||||
ui->peerWidget->setSelectionBehavior(QAbstractItemView::SelectRows);
|
ui->peerWidget->setSelectionBehavior(QAbstractItemView::SelectRows);
|
||||||
ui->peerWidget->setSelectionMode(QAbstractItemView::ExtendedSelection);
|
ui->peerWidget->setSelectionMode(QAbstractItemView::ExtendedSelection);
|
||||||
|
@ -627,10 +628,7 @@ void RPCConsole::setClientModel(ClientModel *model, int bestblock_height, int64_
|
||||||
|
|
||||||
// peer table signal handling - update peer details when selecting new node
|
// peer table signal handling - update peer details when selecting new node
|
||||||
connect(ui->peerWidget->selectionModel(), &QItemSelectionModel::selectionChanged, this, &RPCConsole::updateDetailWidget);
|
connect(ui->peerWidget->selectionModel(), &QItemSelectionModel::selectionChanged, this, &RPCConsole::updateDetailWidget);
|
||||||
// peer table signal handling - update peer details when new nodes are added to the model
|
connect(model->getPeerTableModel(), &PeerTableModel::layoutChanged, this, &RPCConsole::updateDetailWidget);
|
||||||
connect(model->getPeerTableModel(), &PeerTableModel::layoutChanged, this, &RPCConsole::peerLayoutChanged);
|
|
||||||
// peer table signal handling - cache selected node ids
|
|
||||||
connect(model->getPeerTableModel(), &PeerTableModel::layoutAboutToBeChanged, this, &RPCConsole::peerLayoutAboutToChange);
|
|
||||||
|
|
||||||
// set up ban table
|
// set up ban table
|
||||||
ui->banlistWidget->setModel(model->getBanTableModel());
|
ui->banlistWidget->setModel(model->getBanTableModel());
|
||||||
|
@ -1014,67 +1012,6 @@ void RPCConsole::updateTrafficStats(quint64 totalBytesIn, quint64 totalBytesOut)
|
||||||
ui->lblBytesOut->setText(GUIUtil::formatBytes(totalBytesOut));
|
ui->lblBytesOut->setText(GUIUtil::formatBytes(totalBytesOut));
|
||||||
}
|
}
|
||||||
|
|
||||||
void RPCConsole::peerLayoutAboutToChange()
|
|
||||||
{
|
|
||||||
cachedNodeids.clear();
|
|
||||||
for (const QModelIndex& peer : GUIUtil::getEntryData(ui->peerWidget, PeerTableModel::NetNodeId)) {
|
|
||||||
const auto stats = peer.data(PeerTableModel::StatsRole).value<CNodeCombinedStats*>();
|
|
||||||
cachedNodeids.append(stats->nodeStats.nodeid);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void RPCConsole::peerLayoutChanged()
|
|
||||||
{
|
|
||||||
if (!clientModel || !clientModel->getPeerTableModel())
|
|
||||||
return;
|
|
||||||
|
|
||||||
bool fUnselect = false;
|
|
||||||
bool fReselect = false;
|
|
||||||
|
|
||||||
if (cachedNodeids.empty()) // no node selected yet
|
|
||||||
return;
|
|
||||||
|
|
||||||
// find the currently selected row
|
|
||||||
int selectedRow = -1;
|
|
||||||
QModelIndexList selectedModelIndex = ui->peerWidget->selectionModel()->selectedIndexes();
|
|
||||||
if (!selectedModelIndex.isEmpty()) {
|
|
||||||
selectedRow = selectedModelIndex.first().row();
|
|
||||||
}
|
|
||||||
|
|
||||||
// check if our detail node has a row in the table (it may not necessarily
|
|
||||||
// be at selectedRow since its position can change after a layout change)
|
|
||||||
int detailNodeRow = clientModel->getPeerTableModel()->getRowByNodeId(cachedNodeids.first());
|
|
||||||
|
|
||||||
if (detailNodeRow < 0)
|
|
||||||
{
|
|
||||||
// detail node disappeared from table (node disconnected)
|
|
||||||
fUnselect = true;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (detailNodeRow != selectedRow)
|
|
||||||
{
|
|
||||||
// detail node moved position
|
|
||||||
fUnselect = true;
|
|
||||||
fReselect = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (fUnselect && selectedRow >= 0) {
|
|
||||||
clearSelectedNode();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (fReselect)
|
|
||||||
{
|
|
||||||
for(int i = 0; i < cachedNodeids.size(); i++)
|
|
||||||
{
|
|
||||||
ui->peerWidget->selectRow(clientModel->getPeerTableModel()->getRowByNodeId(cachedNodeids.at(i)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
updateDetailWidget();
|
|
||||||
}
|
|
||||||
|
|
||||||
void RPCConsole::updateDetailWidget()
|
void RPCConsole::updateDetailWidget()
|
||||||
{
|
{
|
||||||
const QList<QModelIndex> selected_peers = GUIUtil::getEntryData(ui->peerWidget, PeerTableModel::NetNodeId);
|
const QList<QModelIndex> selected_peers = GUIUtil::getEntryData(ui->peerWidget, PeerTableModel::NetNodeId);
|
||||||
|
|
|
@ -118,10 +118,6 @@ public Q_SLOTS:
|
||||||
void browseHistory(int offset);
|
void browseHistory(int offset);
|
||||||
/** Scroll console view to end */
|
/** Scroll console view to end */
|
||||||
void scrollToEnd();
|
void scrollToEnd();
|
||||||
/** Handle selection caching before update */
|
|
||||||
void peerLayoutAboutToChange();
|
|
||||||
/** Handle updated peer information */
|
|
||||||
void peerLayoutChanged();
|
|
||||||
/** Disconnect a selected node on the Peers tab */
|
/** Disconnect a selected node on the Peers tab */
|
||||||
void disconnectSelectedNode();
|
void disconnectSelectedNode();
|
||||||
/** Ban a selected node on the Peers tab */
|
/** Ban a selected node on the Peers tab */
|
||||||
|
|
Loading…
Add table
Reference in a new issue