mirror of
https://github.com/Retropex/bitcoin.git
synced 2025-05-13 03:30:42 +02:00
GUI: Add dialog to visualise blocks & templates
This commit is contained in:
parent
4505d469ec
commit
2292675c02
@ -46,6 +46,7 @@ QT_MOC_CPP = \
|
||||
qt/moc_bitcoinamountfield.cpp \
|
||||
qt/moc_bitcoingui.cpp \
|
||||
qt/moc_bitcoinunits.cpp \
|
||||
qt/moc_blockview.cpp \
|
||||
qt/moc_clientmodel.cpp \
|
||||
qt/moc_coincontroldialog.cpp \
|
||||
qt/moc_coincontroltreewidget.cpp \
|
||||
@ -118,6 +119,7 @@ BITCOIN_QT_H = \
|
||||
qt/bitcoinamountfield.h \
|
||||
qt/bitcoingui.h \
|
||||
qt/bitcoinunits.h \
|
||||
qt/blockview.h \
|
||||
qt/clientmodel.h \
|
||||
qt/coincontroldialog.h \
|
||||
qt/coincontroltreewidget.h \
|
||||
@ -231,6 +233,7 @@ BITCOIN_QT_BASE_CPP = \
|
||||
qt/bitcoinamountfield.cpp \
|
||||
qt/bitcoingui.cpp \
|
||||
qt/bitcoinunits.cpp \
|
||||
qt/blockview.cpp \
|
||||
qt/clientmodel.cpp \
|
||||
qt/csvmodelwriter.cpp \
|
||||
qt/guiutil.cpp \
|
||||
|
@ -7,6 +7,7 @@
|
||||
#include <qt/bitcoingui.h>
|
||||
|
||||
#include <qt/bitcoinunits.h>
|
||||
#include <qt/blockview.h>
|
||||
#include <qt/clientmodel.h>
|
||||
#include <qt/createwalletdialog.h>
|
||||
#include <qt/guiconstants.h>
|
||||
@ -586,6 +587,14 @@ void BitcoinGUI::createMenuBar()
|
||||
window_menu->addAction(m_show_netwatch_action);
|
||||
window_menu->addAction(showMempoolStatsAction);
|
||||
|
||||
auto show_blockview_action = new QAction(tr("Block &Visualizer"), this);
|
||||
window_menu->addAction(show_blockview_action);
|
||||
connect(show_blockview_action, &QAction::triggered, [this] {
|
||||
auto blockview = new GuiBlockView(platformStyle, m_network_style);
|
||||
blockview->setClientModel(clientModel);
|
||||
GUIUtil::bringToFront(blockview);
|
||||
});
|
||||
|
||||
window_menu->addSeparator();
|
||||
for (RPCConsole::TabTypes tab_type : rpcConsole->tabs()) {
|
||||
QAction* tab_action = window_menu->addAction(rpcConsole->tabTitle(tab_type));
|
||||
|
513
src/qt/blockview.cpp
Normal file
513
src/qt/blockview.cpp
Normal file
@ -0,0 +1,513 @@
|
||||
// Copyright (c) 2024 Luke Dashjr
|
||||
// Distributed under the MIT software license, see the accompanying
|
||||
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
||||
|
||||
#if defined(HAVE_CONFIG_H)
|
||||
#include <config/bitcoin-config.h>
|
||||
#endif
|
||||
|
||||
#include <qt/blockview.h>
|
||||
|
||||
#include <interfaces/node.h>
|
||||
#include <logging.h>
|
||||
#include <node/context.h>
|
||||
#include <node/miner.h>
|
||||
#include <primitives/block.h>
|
||||
#include <util/strencodings.h>
|
||||
#include <validation.h>
|
||||
#include <validationinterface.h>
|
||||
|
||||
#include <qt/bitcoinunits.h>
|
||||
#include <qt/clientmodel.h>
|
||||
#include <qt/guiutil.h>
|
||||
#include <qt/networkstyle.h>
|
||||
#include <qt/optionsmodel.h>
|
||||
|
||||
#include <cmath>
|
||||
#include <numbers>
|
||||
|
||||
#include <QComboBox>
|
||||
#include <QLabel>
|
||||
#include <QGraphicsEllipseItem>
|
||||
#include <QGraphicsScene>
|
||||
#include <QGraphicsView>
|
||||
#include <QHBoxLayout>
|
||||
#include <QVBoxLayout>
|
||||
|
||||
static constexpr qreal TX_PADDING_NEXT{4};
|
||||
static constexpr qreal TX_PADDING_NEARBY{2};
|
||||
static constexpr qreal EXPECTED_WHITESPACE_PERCENT{1.5};
|
||||
static constexpr auto RADIAN_DIVISOR{8};
|
||||
|
||||
void ScalingGraphicsView::resizeEvent(QResizeEvent * const event)
|
||||
{
|
||||
fitInView(scene()->sceneRect(), Qt::KeepAspectRatio);
|
||||
QGraphicsView::resizeEvent(event);
|
||||
}
|
||||
|
||||
class BlockViewValidationInterface final : public CValidationInterface
|
||||
{
|
||||
private:
|
||||
GuiBlockView& m_bv;
|
||||
|
||||
public:
|
||||
explicit BlockViewValidationInterface(GuiBlockView& bv) : m_bv(bv) {}
|
||||
|
||||
void BlockConnected(ChainstateRole role, const std::shared_ptr<const CBlock>& block_cached, const CBlockIndex* pblockindex) override {
|
||||
m_bv.updateBestBlock(pblockindex->nHeight);
|
||||
|
||||
if (!m_bv.m_follow_tip) return;
|
||||
|
||||
std::shared_ptr<const CBlock> block = block_cached;
|
||||
auto chainman = m_bv.getChainstateManager();
|
||||
Assert(chainman);
|
||||
if (!block) {
|
||||
std::shared_ptr<CBlock> pblock = std::make_shared<CBlock>();
|
||||
if (!chainman->m_blockman.ReadBlockFromDisk(*pblock, *pblockindex)) {
|
||||
// Indicate error somehow?
|
||||
return;
|
||||
}
|
||||
block = pblock;
|
||||
}
|
||||
|
||||
const auto block_subsidy = GetBlockSubsidy(pblockindex->nHeight, chainman->GetParams().GetConsensus());
|
||||
|
||||
m_bv.setBlock(block, block_subsidy);
|
||||
}
|
||||
|
||||
void NewBlockTemplate(const std::shared_ptr<node::CBlockTemplate>& blocktemplate) override {
|
||||
{
|
||||
LOCK(m_bv.m_mutex);
|
||||
if (m_bv.m_block) {
|
||||
// Update cached template, but don't render it
|
||||
m_bv.m_block_template = blocktemplate;
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
m_bv.setBlock(blocktemplate);
|
||||
}
|
||||
};
|
||||
|
||||
void GuiBlockView::updateBestBlock(const int height)
|
||||
{
|
||||
m_block_chooser->setItemText(1, tr("Newest block (%1)").arg(height));
|
||||
}
|
||||
|
||||
GuiBlockView::GuiBlockView(const PlatformStyle *platformStyle, const NetworkStyle *networkStyle, QWidget *parent) :
|
||||
QDialog(parent, GUIUtil::dialog_flags | Qt::WindowMaximizeButtonHint)
|
||||
{
|
||||
setWindowTitle(tr(PACKAGE_NAME) + " - " + tr("Block View") + " " + networkStyle->getTitleAddText());
|
||||
setWindowIcon(networkStyle->getTrayAndWindowIcon());
|
||||
resize(640, 640);
|
||||
|
||||
QVBoxLayout * const layout = new QVBoxLayout(this);
|
||||
setLayout(layout);
|
||||
|
||||
auto hlayout = new QHBoxLayout;
|
||||
layout->addLayout(hlayout);
|
||||
hlayout->addWidget(new QLabel(tr("Displayed block: ")));
|
||||
m_block_chooser = new QComboBox(this);
|
||||
hlayout->addWidget(m_block_chooser, 1);
|
||||
connect(m_block_chooser, QOverload<int>::of(&QComboBox::currentIndexChanged), [=, this](const int index){
|
||||
m_follow_tip = false;
|
||||
auto ud = m_block_chooser->itemData(index).toInt();
|
||||
if (ud == -3) {
|
||||
m_block_chooser->setEditable(false);
|
||||
auto block_template = WITH_LOCK(m_mutex, return m_block_template);
|
||||
if (block_template) {
|
||||
setBlock(block_template);
|
||||
} else {
|
||||
clear();
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
auto chainman = getChainstateManager();
|
||||
if (!chainman) {
|
||||
clear();
|
||||
return;
|
||||
}
|
||||
auto& blockman = chainman->m_blockman;
|
||||
|
||||
CBlockIndex *pblockindex;
|
||||
if (ud == -2) {
|
||||
m_follow_tip = true;
|
||||
pblockindex = WITH_LOCK(::cs_main, return chainman->ActiveChain().Tip());
|
||||
if (!pblockindex) {
|
||||
clear();
|
||||
return;
|
||||
}
|
||||
} else if (ud == -1) {
|
||||
m_block_chooser->setEditable(true);
|
||||
m_block_chooser->clearEditText();
|
||||
return;
|
||||
} else {
|
||||
auto qtxt = m_block_chooser->itemText(index);
|
||||
auto txt = qtxt.toStdString();
|
||||
auto blockhash{uint256::FromHex(txt)};
|
||||
if (blockhash) {
|
||||
LOCK(cs_main);
|
||||
pblockindex = blockman.LookupBlockIndex(*blockhash);
|
||||
} else if (auto height = ToIntegral<int>(txt)) {
|
||||
LOCK(cs_main);
|
||||
pblockindex = chainman->ActiveChain()[*height];
|
||||
} else {
|
||||
pblockindex = nullptr;
|
||||
}
|
||||
if (!pblockindex) {
|
||||
clear();
|
||||
QMessageBox::critical(this, tr("Invalid block"), tr("\"%1\" is not a valid block height or hash!").arg(qtxt));
|
||||
m_block_chooser->removeItem(index);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
std::shared_ptr<CBlock> block = std::make_shared<CBlock>();
|
||||
if ((!blockman.ReadBlockFromDisk(*block, *pblockindex)) || block->vtx.empty()) {
|
||||
clear();
|
||||
const bool is_pruned = WITH_LOCK(::cs_main, return blockman.IsBlockPruned(*pblockindex));
|
||||
if (is_pruned) {
|
||||
QMessageBox::critical(this, tr("Pruned block"), tr("Block %1 (%2) is pruned.").arg(pblockindex->nHeight).arg(QString::fromStdString(pblockindex->GetBlockHash().ToString())));
|
||||
} else {
|
||||
QMessageBox::critical(this, tr("Error reading block"), tr("Block %1 (%2) could not be loaded.").arg(pblockindex->nHeight).arg(QString::fromStdString(pblockindex->GetBlockHash().ToString())));
|
||||
}
|
||||
m_block_chooser->removeItem(index);
|
||||
return;
|
||||
}
|
||||
|
||||
m_block_chooser->setEditable(false);
|
||||
|
||||
const auto block_subsidy = GetBlockSubsidy(pblockindex->nHeight, chainman->GetParams().GetConsensus());
|
||||
|
||||
setBlock(block, block_subsidy);
|
||||
});
|
||||
// Items initialized later, after ClientModel is available
|
||||
|
||||
m_scene = new QGraphicsScene(this);
|
||||
m_scene->setSceneRect(0, 0, 1, 1);
|
||||
auto view = new ScalingGraphicsView(m_scene, this);
|
||||
layout->addWidget(view);
|
||||
view->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
||||
view->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
||||
view->setStyleSheet("QGraphicsView { background: transparent; }");
|
||||
view->setAlignment(Qt::AlignHCenter | Qt::AlignBottom);
|
||||
connect(m_scene, &QGraphicsScene::sceneRectChanged, [view](const QRectF& rect){
|
||||
view->fitInView(rect, Qt::KeepAspectRatio);
|
||||
});
|
||||
|
||||
hlayout = new QHBoxLayout;
|
||||
layout->addLayout(hlayout);
|
||||
hlayout->addWidget(new QLabel(tr("Transactions"), this));
|
||||
m_lbl_tx_count = new QLabel(this);
|
||||
m_lbl_tx_count->setAlignment(Qt::AlignRight);
|
||||
hlayout->addWidget(m_lbl_tx_count);
|
||||
|
||||
hlayout = new QHBoxLayout;
|
||||
layout->addLayout(hlayout);
|
||||
hlayout->addWidget(new QLabel(tr("Txn Fees"), this));
|
||||
m_lbl_tx_fees = new QLabel(this);
|
||||
m_lbl_tx_fees->setAlignment(Qt::AlignRight);
|
||||
hlayout->addWidget(m_lbl_tx_fees);
|
||||
|
||||
connect(&m_timer, &QTimer::timeout, this, &GuiBlockView::updateScene);
|
||||
|
||||
m_validation_interface = new BlockViewValidationInterface(*this);
|
||||
}
|
||||
|
||||
GuiBlockView::~GuiBlockView()
|
||||
{
|
||||
if (m_validation_interface) {
|
||||
setClientModel(nullptr);
|
||||
delete m_validation_interface;
|
||||
m_validation_interface = nullptr;
|
||||
}
|
||||
}
|
||||
|
||||
void GuiBlockView::setClientModel(ClientModel *model)
|
||||
{
|
||||
if (m_client_model) {
|
||||
auto& validation_signals = m_client_model->node().context()->validation_signals;
|
||||
if (validation_signals) {
|
||||
validation_signals->UnregisterValidationInterface(m_validation_interface);
|
||||
}
|
||||
disconnect(m_client_model->getOptionsModel(), &OptionsModel::displayUnitChanged, this, &GuiBlockView::updateDisplayUnit);
|
||||
}
|
||||
m_client_model = model;
|
||||
if (model) {
|
||||
connect(model->getOptionsModel(), &OptionsModel::displayUnitChanged, this, &GuiBlockView::updateDisplayUnit);
|
||||
updateDisplayUnit();
|
||||
|
||||
if (m_block_chooser->count() == 0) {
|
||||
m_block_chooser->addItem(tr("This node's preferred block template"), -3);
|
||||
m_block_chooser->addItem("", -2);
|
||||
m_block_chooser->addItem(tr("Specific block"), -1);
|
||||
m_block_chooser->setCurrentIndex(1);
|
||||
}
|
||||
|
||||
auto chainman = getChainstateManager();
|
||||
if (chainman) {
|
||||
const auto pblockindex = WITH_LOCK(::cs_main, return chainman->ActiveChain().Tip());
|
||||
updateBestBlock(pblockindex->nHeight);
|
||||
}
|
||||
auto& validation_signals = model->node().context()->validation_signals;
|
||||
if (validation_signals) {
|
||||
validation_signals->RegisterValidationInterface(m_validation_interface);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
ChainstateManager* GuiBlockView::getChainstateManager() const
|
||||
{
|
||||
if (!m_client_model) return nullptr;
|
||||
auto node_ctx = m_client_model->node().context();
|
||||
if (!node_ctx) return nullptr;
|
||||
auto& chainman = node_ctx->chainman;
|
||||
if (!chainman) return nullptr;
|
||||
return &(*chainman);
|
||||
}
|
||||
|
||||
void GuiBlockView::clear()
|
||||
{
|
||||
LOCK(m_mutex);
|
||||
m_block_fees = -1;
|
||||
m_lbl_tx_count->setText("");
|
||||
m_block.reset();
|
||||
m_block_template.reset();
|
||||
for (auto& [wtxid, elem] : m_elements) {
|
||||
const auto gi = elem.gi;
|
||||
m_scene->removeItem(gi);
|
||||
delete gi;
|
||||
}
|
||||
m_elements.clear();
|
||||
}
|
||||
|
||||
bool GuiBlockView::any_overlap(const Bubble& proposed, const std::vector<Bubble>& others)
|
||||
{
|
||||
for (const auto& other : others) {
|
||||
const auto x_dist = std::abs(other.pos.x() - proposed.pos.x());
|
||||
const auto y_dist = std::abs(other.pos.y() - proposed.pos.y());
|
||||
const auto dist = std::sqrt((x_dist * x_dist) + (y_dist * y_dist));
|
||||
if (dist < proposed.radius + other.radius + TX_PADDING_NEARBY) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
void GuiBlockView::setBlock(std::shared_ptr<const CBlock> block, const CAmount block_subsidy)
|
||||
{
|
||||
LOCK(m_mutex);
|
||||
m_block_fees = [&] {
|
||||
CAmount total{0};
|
||||
Assert(!block->vtx.empty());
|
||||
for (const auto& outp : block->vtx[0]->vout) {
|
||||
total += outp.nValue;
|
||||
}
|
||||
return total - block_subsidy;
|
||||
}();
|
||||
m_block = block;
|
||||
m_block_template.reset();
|
||||
m_block_changed = true;
|
||||
updateElements(/*instant=*/ true);
|
||||
}
|
||||
|
||||
void GuiBlockView::setBlock(std::shared_ptr<const node::CBlockTemplate> blocktemplate)
|
||||
{
|
||||
LOCK(m_mutex);
|
||||
const bool instant = (bool)m_block; // force instant if changing from real block to template
|
||||
m_block_fees = -blocktemplate->vTxFees.front();
|
||||
m_block.reset();
|
||||
m_block_template = blocktemplate;
|
||||
m_block_changed = true;
|
||||
updateElements(/*instant=*/ instant);
|
||||
}
|
||||
|
||||
void GuiBlockView::updateBlockFees(CAmount block_fees)
|
||||
{
|
||||
if (block_fees < 0) {
|
||||
m_lbl_tx_fees->setText("");
|
||||
return;
|
||||
}
|
||||
const auto unit = m_client_model ? m_client_model->getOptionsModel()->getDisplayUnit() : BitcoinUnit::BTC;
|
||||
m_lbl_tx_fees->setText(BitcoinUnits::formatWithUnit(unit, block_fees));
|
||||
}
|
||||
|
||||
void GuiBlockView::updateDisplayUnit()
|
||||
{
|
||||
const auto block_fees = WITH_LOCK(m_mutex, return m_block_fees);
|
||||
updateBlockFees(block_fees);
|
||||
}
|
||||
|
||||
constexpr qreal offscreen{99};
|
||||
|
||||
void GuiBlockView::updateElements(bool instant)
|
||||
{
|
||||
if (!m_block_changed) return;
|
||||
|
||||
m_timer.stop();
|
||||
m_block_changed = false;
|
||||
auto pblocktemplate = m_block_template;
|
||||
auto pblock = m_block;
|
||||
auto& block = pblock ? *pblock : pblocktemplate->block;
|
||||
|
||||
instant |= m_elements.empty();
|
||||
for (auto& el : m_elements) {
|
||||
el.second.target_loc.setY(offscreen);
|
||||
}
|
||||
m_bubblegraph = std::make_unique<BubbleGraph>();
|
||||
auto& bubbles = m_bubblegraph->bubbles;
|
||||
size_t total_txs_size{0};
|
||||
qreal limit_halfwidth{std::sqrt(::GetSerializeSize(TX_WITH_WITNESS(block))) * EXPECTED_WHITESPACE_PERCENT / 2};
|
||||
for (size_t i = 1; i < block.vtx.size(); ++i) {
|
||||
auto& tx = *block.vtx[i];
|
||||
auto& el = m_elements[tx.GetWitnessHash()];
|
||||
QPointF preferred_loc;
|
||||
double diameter;
|
||||
const auto tx_size = tx.GetTotalSize();
|
||||
total_txs_size += tx_size;
|
||||
const bool fresh_bubble = !el.gi;
|
||||
if (fresh_bubble) {
|
||||
diameter = 2 * std::sqrt(tx_size / std::numbers::pi);
|
||||
} else {
|
||||
// preferred_loc = el.gi->pos();
|
||||
diameter = el.gi->boundingRect().height();
|
||||
}
|
||||
Bubble proposed{ .pos = {}, .radius = diameter / 2, .el = &el, };
|
||||
qreal x_extremity{proposed.radius};
|
||||
if (bubbles.empty()) {
|
||||
proposed.pos.setY(-proposed.radius);
|
||||
}
|
||||
for (auto bubble_it = bubbles.rbegin(); bubble_it != bubbles.rend(); ++bubble_it) {
|
||||
const auto& centre = bubble_it->pos;
|
||||
QPointF preferred_loc_rel(preferred_loc.x() - centre.x(), preferred_loc.y() - centre.y());
|
||||
double preferred_angle;
|
||||
if (preferred_loc_rel.isNull()) {
|
||||
preferred_angle = std::numbers::pi / 2;
|
||||
} else {
|
||||
preferred_angle = std::atan2(preferred_loc.y() - centre.y(), preferred_loc.x() - centre.x());
|
||||
}
|
||||
const auto distance = bubble_it->radius + proposed.radius + TX_PADDING_NEXT;
|
||||
double angle = preferred_angle;
|
||||
bool found{false};
|
||||
while (true) {
|
||||
proposed.pos = QPointF(centre.x() + (distance * std::cos(angle)), centre.y() + (distance * std::sin(angle)));
|
||||
|
||||
x_extremity = std::abs(proposed.pos.x()) + proposed.radius;
|
||||
if (proposed.pos.y() < -proposed.radius && x_extremity <= limit_halfwidth && !any_overlap(proposed, bubbles)) {
|
||||
found = true;
|
||||
break;
|
||||
}
|
||||
|
||||
if (angle < preferred_angle) {
|
||||
angle = preferred_angle + (preferred_angle - angle);
|
||||
} else {
|
||||
angle = preferred_angle - (angle - preferred_angle) - (std::numbers::pi / RADIAN_DIVISOR);
|
||||
}
|
||||
if (angle > preferred_angle + std::numbers::pi) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (found) break;
|
||||
}
|
||||
m_bubblegraph->min_x = std::min(m_bubblegraph->min_x, proposed.pos.x() - proposed.radius);
|
||||
m_bubblegraph->max_x = std::max(m_bubblegraph->max_x, proposed.pos.x() + proposed.radius);
|
||||
m_bubblegraph->min_y = std::min(m_bubblegraph->min_y, proposed.pos.y() - proposed.radius);
|
||||
bubbles.push_back(proposed);
|
||||
el.target_loc = proposed.pos;
|
||||
}
|
||||
m_lbl_tx_count->setText(tr("%1 (%2)").arg(block.vtx.size() - 1).arg(tr("%1 kB").arg(total_txs_size / 1000.0, 0, 'f', 1)));
|
||||
updateBlockFees(m_block_fees);
|
||||
m_bubblegraph->instant = instant;
|
||||
QMetaObject::invokeMethod(this, "updateSceneInit", Qt::QueuedConnection);
|
||||
}
|
||||
|
||||
void GuiBlockView::updateSceneInit()
|
||||
{
|
||||
LOCK(m_mutex);
|
||||
if (!m_bubblegraph) return;
|
||||
for (auto& bubble : m_bubblegraph->bubbles) {
|
||||
auto& el = *bubble.el;
|
||||
if (!el.gi) {
|
||||
const auto diameter = bubble.radius * 2;
|
||||
auto gi = m_scene->addEllipse(0, 0, diameter, diameter, QPen(palette().window(), TX_PADDING_NEARBY));
|
||||
el.gi = gi;
|
||||
gi->setBrush(QColor(Qt::blue));
|
||||
gi->setPos(bubble.pos.x() - bubble.radius, m_bubblegraph->instant ? (bubble.pos.y() - bubble.radius) : offscreen);
|
||||
}
|
||||
}
|
||||
for (auto it = m_elements.begin(); it != m_elements.end(); ) {
|
||||
const auto& target_loc = it->second.target_loc;
|
||||
const auto gi = it->second.gi;
|
||||
bool delete_el{false};
|
||||
if (target_loc.y() == offscreen || !gi /* never got a chance to exist */) {
|
||||
delete_el = true;
|
||||
// TODO: if confirmed, slide it off the bottom
|
||||
// TODO: if conflicted, pop the bubble?
|
||||
// TODO: if delayed, move off the top
|
||||
} else {
|
||||
if (gi->y() == offscreen) {
|
||||
gi->setY(m_bubblegraph->min_y - gi->boundingRect().height());
|
||||
}
|
||||
}
|
||||
if (delete_el) {
|
||||
if (gi) {
|
||||
m_scene->removeItem(gi);
|
||||
delete gi;
|
||||
}
|
||||
it = m_elements.erase(it);
|
||||
} else {
|
||||
++it;
|
||||
}
|
||||
}
|
||||
m_scene->setSceneRect(m_bubblegraph->min_x, m_bubblegraph->min_y, m_bubblegraph->max_x - m_bubblegraph->min_x, -m_bubblegraph->min_y);
|
||||
if (!m_bubblegraph->instant) {
|
||||
m_frame_div = 4;
|
||||
updateScene();
|
||||
m_timer.start(100);
|
||||
}
|
||||
m_bubblegraph.reset();
|
||||
}
|
||||
|
||||
void GuiBlockView::updateScene()
|
||||
{
|
||||
LOCK(m_mutex);
|
||||
bool all_completed{true};
|
||||
for (auto it = m_elements.begin(); it != m_elements.end(); ) {
|
||||
const auto& target_loc = it->second.target_loc;
|
||||
QGraphicsItem* gi = it->second.gi;
|
||||
const auto radius = gi->boundingRect().width() / 2;
|
||||
const QPointF current_loc(gi->pos().x() + radius, gi->pos().y() + radius);
|
||||
bool delete_el{false};
|
||||
if (target_loc != current_loc) {
|
||||
// Get 25% closer each tick
|
||||
QPointF new_loc(current_loc.x() + ((target_loc.x() - current_loc.x()) / m_frame_div),
|
||||
current_loc.y() + ((target_loc.y() - current_loc.y()) / m_frame_div));
|
||||
if (std::abs(new_loc.x() - target_loc.x()) < TX_PADDING_NEXT) {
|
||||
new_loc.setX(target_loc.x());
|
||||
}
|
||||
if (std::abs(new_loc.y() - target_loc.y()) < TX_PADDING_NEXT) {
|
||||
new_loc.setY(target_loc.y());
|
||||
}
|
||||
gi->setPos(new_loc.x() - radius, new_loc.y() - radius);
|
||||
if (new_loc == target_loc) {
|
||||
if (target_loc.y() + radius < m_scene->sceneRect().y() || target_loc.y() - radius > 0) {
|
||||
delete_el = true;
|
||||
}
|
||||
} else {
|
||||
all_completed = false;
|
||||
}
|
||||
}
|
||||
if (delete_el) {
|
||||
m_scene->removeItem(gi);
|
||||
delete gi;
|
||||
it = m_elements.erase(it);
|
||||
} else {
|
||||
++it;
|
||||
}
|
||||
}
|
||||
--m_frame_div;
|
||||
if (all_completed) {
|
||||
m_timer.stop();
|
||||
}
|
||||
}
|
120
src/qt/blockview.h
Normal file
120
src/qt/blockview.h
Normal file
@ -0,0 +1,120 @@
|
||||
// Copyright (c) 2024 Luke Dashjr
|
||||
// Distributed under the MIT software license, see the accompanying
|
||||
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
||||
|
||||
#ifndef BITCOIN_QT_BLOCKVIEW_H
|
||||
#define BITCOIN_QT_BLOCKVIEW_H
|
||||
|
||||
#include <consensus/amount.h>
|
||||
#include <sync.h>
|
||||
#include <threadsafety.h>
|
||||
#include <util/transaction_identifier.h>
|
||||
|
||||
#include <atomic>
|
||||
#include <map>
|
||||
#include <memory>
|
||||
#include <vector>
|
||||
|
||||
#include <QDialog>
|
||||
#include <QGraphicsView>
|
||||
#include <QPointF>
|
||||
#include <QTimer>
|
||||
|
||||
QT_BEGIN_NAMESPACE
|
||||
class QComboBox;
|
||||
class QGraphicsItem;
|
||||
class QGraphicsScene;
|
||||
class QLabel;
|
||||
class QWidget;
|
||||
QT_END_NAMESPACE
|
||||
|
||||
class CBlock;
|
||||
namespace node { struct CBlockTemplate; }
|
||||
class ChainstateManager;
|
||||
class ClientModel;
|
||||
class CValidationInterface;
|
||||
class NetworkStyle;
|
||||
class PlatformStyle;
|
||||
|
||||
class ScalingGraphicsView : public QGraphicsView
|
||||
{
|
||||
Q_OBJECT
|
||||
|
||||
public:
|
||||
using QGraphicsView::QGraphicsView;
|
||||
|
||||
void resizeEvent(QResizeEvent *event) override;
|
||||
};
|
||||
|
||||
class BlockViewValidationInterface;
|
||||
|
||||
class GuiBlockView : public QDialog
|
||||
{
|
||||
Q_OBJECT
|
||||
|
||||
public:
|
||||
RecursiveMutex m_mutex;
|
||||
|
||||
private:
|
||||
struct SceneElement {
|
||||
QGraphicsItem* gi;
|
||||
QPointF target_loc;
|
||||
};
|
||||
struct Bubble {
|
||||
QPointF pos;
|
||||
double radius;
|
||||
SceneElement *el;
|
||||
};
|
||||
struct BubbleGraph {
|
||||
std::vector<Bubble> bubbles;
|
||||
qreal min_x{0};
|
||||
qreal max_x{0};
|
||||
qreal min_y{0};
|
||||
bool instant;
|
||||
};
|
||||
std::map<Wtxid, SceneElement> m_elements GUARDED_BY(m_mutex);
|
||||
std::unique_ptr<BubbleGraph> m_bubblegraph GUARDED_BY(m_mutex);
|
||||
QGraphicsScene *m_scene;
|
||||
QTimer m_timer;
|
||||
unsigned int m_frame_div;
|
||||
|
||||
QComboBox *m_block_chooser;
|
||||
QLabel *m_lbl_tx_count;
|
||||
QLabel *m_lbl_tx_fees;
|
||||
|
||||
BlockViewValidationInterface *m_validation_interface;
|
||||
|
||||
static bool any_overlap(const Bubble& proposed, const std::vector<Bubble>& others);
|
||||
|
||||
protected:
|
||||
void updateElements(bool instant) EXCLUSIVE_LOCKS_REQUIRED(m_mutex);
|
||||
void updateBlockFees(CAmount block_fees);
|
||||
|
||||
protected Q_SLOTS:
|
||||
void updateDisplayUnit();
|
||||
void updateSceneInit();
|
||||
void updateScene();
|
||||
|
||||
public:
|
||||
ClientModel *m_client_model{nullptr};
|
||||
|
||||
bool m_block_changed GUARDED_BY(m_mutex);
|
||||
CAmount m_block_fees GUARDED_BY(m_mutex) {-1};
|
||||
std::shared_ptr<const node::CBlockTemplate> m_block_template GUARDED_BY(m_mutex);
|
||||
std::shared_ptr<const CBlock> m_block GUARDED_BY(m_mutex);
|
||||
std::atomic<bool> m_follow_tip;
|
||||
|
||||
GuiBlockView(const PlatformStyle *, const NetworkStyle *, QWidget * parent = nullptr);
|
||||
~GuiBlockView();
|
||||
|
||||
void setClientModel(ClientModel *model);
|
||||
ChainstateManager* getChainstateManager() const;
|
||||
|
||||
void updateBestBlock(int height);
|
||||
|
||||
void clear();
|
||||
void setBlock(std::shared_ptr<const CBlock> block, CAmount block_subsidy);
|
||||
void setBlock(std::shared_ptr<const node::CBlockTemplate> blocktemplate);
|
||||
};
|
||||
|
||||
#endif // BITCOIN_QT_BLOCKVIEW_H
|
Loading…
Reference in New Issue
Block a user