/* This file is part of cpp-ethereum. cpp-ethereum 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 3 of the License, or (at your option) any later version. cpp-ethereum 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 cpp-ethereum. If not, see . */ /** @file DownloadView.cpp * @author Gav Wood * @date 2014 */ #include "DownloadView.h" #include #include #include #include "Grapher.h" using namespace std; using namespace dev; using namespace az; using namespace eth; SyncView::SyncView(QWidget* _p): QWidget(_p) { } void SyncView::paintEvent(QPaintEvent*) { QPainter painter(this); painter.fillRect(rect(), Qt::white); painter.setRenderHint(QPainter::Antialiasing, true); painter.setRenderHint(QPainter::HighQualityAntialiasing, true); if (!m_client || !isVisible() || !rect().width() || !rect().height()) return; DownloadMan const* man = m_client->downloadMan(); BlockQueueStatus bqs = m_client->blockQueueStatus(); SyncStatus sync = m_client->syncStatus(); unsigned syncFrom = m_client->numberFromHash(PendingBlockHash); unsigned syncImported = syncFrom; unsigned syncImporting = syncImported + bqs.importing; unsigned syncVerified = syncImporting + bqs.verified; unsigned syncVerifying = syncVerified + bqs.verifying; unsigned syncUnverified = syncVerifying + bqs.unverified; unsigned syncCount = syncUnverified + bqs.unknown - syncFrom; // best effort guess. assumes there's no forks. unsigned downloadFrom = sync.state == SyncState::Idle ? m_lastSyncFrom : m_client->numberFromHash(m_client->isKnown(man->firstBlock()) ? man->firstBlock() : PendingBlockHash); unsigned downloadCount = sync.state == SyncState::Idle ? m_lastSyncCount : sync.blocksTotal; unsigned downloadDone = downloadFrom + (sync.state == SyncState::Idle ? m_lastSyncCount : sync.blocksReceived); unsigned downloadPoint = downloadFrom + (sync.state == SyncState::Idle ? m_lastSyncCount : man->overview().lastComplete); unsigned hashFrom = sync.state == SyncState::Hashes ? m_client->numberFromHash(PendingBlockHash) : downloadFrom; unsigned hashCount = sync.state == SyncState::Hashes ? sync.hashesTotal : downloadCount; unsigned hashDone = hashFrom + (sync.state == SyncState::Hashes ? sync.hashesReceived : hashCount); QString labelText = QString("PV%1").arg(sync.protocolVersion); QColor labelBack = QColor::fromHsv(sync.protocolVersion == 60 ? 30 : sync.protocolVersion == 61 ? 120 : 240, 25, 200); QColor labelFore = labelBack.darker(); switch (sync.state) { case SyncState::Hashes: if (!syncCount || !sync.hashesEstimated) { m_lastSyncFrom = min(hashFrom, m_lastSyncFrom); m_lastSyncCount = max(hashFrom + hashCount, m_lastSyncFrom + m_lastSyncCount) - m_lastSyncFrom; m_wasEstimate = sync.hashesEstimated; } break; case SyncState::Blocks: if (m_wasEstimate) { m_lastSyncFrom = downloadFrom; m_lastSyncCount = downloadCount; m_wasEstimate = false; } break; case SyncState::Idle: if (!syncCount) { m_lastSyncFrom = (unsigned)-1; m_lastSyncCount = 0; labelBack = QColor::fromHsv(0, 0, 200); labelFore = Qt::white; labelText = "Idle"; } default: break; } unsigned from = min(min(hashFrom, downloadFrom), min(syncFrom, m_lastSyncFrom)); unsigned count = max(max(hashFrom + hashCount, downloadFrom + downloadCount), max(syncFrom + syncCount, m_lastSyncFrom + m_lastSyncCount)) - from; /* cnote << "Range " << from << "-" << (from + count) << "(" << hashFrom << "+" << hashCount << "," << downloadFrom << "+" << downloadCount << "," << syncFrom << "+" << syncCount << ")"; auto r = [&](unsigned u) { return toString((u - from) * 100 / count) + "%"; }; if (count) { cnote << "Hashes:" << r(hashDone) << " Blocks:" << r(downloadFlank) << r(downloadDone) << r(downloadPoint); cnote << "Importing:" << r(syncFrom) << r(syncImported) << r(syncImporting) << r(syncVerified) << r(syncVerifying) << r(syncUnverified); } */ float const squareSize = min(rect().width(), rect().height()); auto middleRect = [&](float w, float h) { return QRectF(rect().width() / 2 - w / 2, rect().height() / 2 - h / 2, w, h); }; auto middle = [&](float x) { return middleRect(squareSize * x, squareSize * x); }; auto pieProgress = [&](unsigned h, unsigned s, unsigned v, float row, float thickness, unsigned nfrom, unsigned ncount) { auto arcLen = [&](unsigned x) { return x * -5760.f / count; }; auto arcPos = [&](unsigned x) { return int(90 * 16.f + arcLen(x - from)) % 5760; }; painter.setPen(QPen(QColor::fromHsv(h, s, v), squareSize * thickness, Qt::SolidLine, Qt::FlatCap)); painter.setBrush(Qt::NoBrush); painter.drawArc(middle(0.5 + row / 2), arcPos(nfrom), arcLen(ncount)); }; auto pieProgress2 = [&](unsigned h, unsigned s, unsigned v, float row, float orbit, float thickness, unsigned nfrom, unsigned ncount) { pieProgress(h, s, v, row - orbit, thickness, nfrom, ncount); pieProgress(h, s, v, row + orbit, thickness, nfrom, ncount); }; auto pieLabel = [&](QString text, float points, QColor fore, QColor back) { painter.setBrush(QBrush(back)); painter.setFont(QFont("Helvetica", points, QFont::Bold)); QRectF r = painter.boundingRect(middle(1.f), Qt::AlignCenter, text); r.adjust(-r.width() / 4, -r.height() / 8, r.width() / 4, r.height() / 8); painter.setPen(QPen(fore, r.height() / 20)); painter.drawRoundedRect(r, r.height() / 4, r.height() / 4); painter.drawText(r, Qt::AlignCenter, text); }; float lineHeight = painter.boundingRect(rect(), Qt::AlignTop | Qt::AlignHCenter, "Ay").height(); auto hProgress = [&](unsigned h, unsigned s, unsigned v, float row, float thickness, unsigned nfrom, unsigned ncount) { QRectF r = rect(); painter.setPen(QPen(QColor::fromHsv(h, s, v), r.height() * thickness * 3, Qt::SolidLine, Qt::FlatCap)); painter.setBrush(Qt::NoBrush); auto y = row * (r.height() - lineHeight) + lineHeight; painter.drawLine(QPointF((nfrom - from) * r.width() / count, y), QPointF((nfrom + ncount - from) * r.width() / count, y)); }; auto hProgress2 = [&](unsigned h, unsigned s, unsigned v, float row, float orbit, float thickness, unsigned nfrom, unsigned ncount) { hProgress(h, s, v, row - orbit * 3, thickness, nfrom, ncount); hProgress(h, s, v, row + orbit * 3, thickness, nfrom, ncount); }; auto hLabel = [&](QString text, float points, QColor fore, QColor back) { painter.setBrush(QBrush(back)); painter.setFont(QFont("Helvetica", points, QFont::Bold)); QRectF r = painter.boundingRect(rect(), Qt::AlignTop | Qt::AlignHCenter, text); r.adjust(-r.width() / 4, r.height() / 8, r.width() / 4, 3 * r.height() / 8); painter.setPen(QPen(fore, r.height() / 20)); painter.drawRoundedRect(r, r.height() / 4, r.height() / 4); painter.drawText(r, Qt::AlignCenter, text); }; function progress; function progress2; function label; if (rect().width() / rect().height() > 5) { progress = hProgress; progress2 = hProgress2; label = hLabel; } else { progress = pieProgress; progress2 = pieProgress2; label = pieLabel; } if (sync.state != SyncState::Idle) { progress(0, 0, 220, 0.4f, 0.02f, from, hashDone - from); // Download rail progress(240, 25, 170, 0.4f, 0.02f, downloadDone, downloadPoint - downloadDone); // Latest download point progress(240, 50, 120, 0.4f, 0.04f, from, downloadDone - from); // Downloaded } progress(0, 0, 220, 0.8f, 0.01f, from, count); // Sync rail progress(0, 0, 170, 0.8f, 0.02f, from, syncUnverified - from); // Verification rail progress2(60, 25, 170, 0.8f, 0.06f, 0.005f, from, syncVerifying - from); // Verifying. progress2(120, 25, 170, 0.8f, 0.06f, 0.005f, from, syncVerified - from); // Verified. progress(120, 50, 120, 0.8f, 0.05f, from, syncFrom - from); // Imported. progress(0, 0, 120, 0.8f, 0.02f, syncFrom, syncImporting - syncFrom); // Importing. if (sync.state != SyncState::Idle || (sync.state == SyncState::Idle && !syncCount)) label(labelText, 11, labelFore, labelBack); }