Files
Jungfraujoch/viewer/widgets/ToolbarIcons.cpp
T
leonarski_f 75e401f0e5
Build Packages / Unit tests (push) Successful in 1h31m59s
Build Packages / build:rpm (rocky8_nocuda) (push) Successful in 8m43s
Build Packages / build:rpm (rocky9_nocuda) (push) Successful in 10m5s
Build Packages / build:rpm (ubuntu2204_nocuda) (push) Successful in 9m27s
Build Packages / build:rpm (ubuntu2404_nocuda) (push) Successful in 8m56s
Build Packages / build:rpm (rocky8_sls9) (push) Successful in 9m24s
Build Packages / build:rpm (rocky9_sls9) (push) Successful in 10m27s
Build Packages / build:rpm (rocky8) (push) Successful in 9m20s
Build Packages / build:rpm (rocky9) (push) Successful in 10m50s
Build Packages / build:rpm (ubuntu2204) (push) Successful in 9m54s
Build Packages / build:rpm (ubuntu2404) (push) Successful in 8m38s
Build Packages / DIALS test (push) Successful in 12m13s
Build Packages / XDS test (durin plugin) (push) Successful in 7m8s
Build Packages / XDS test (JFJoch plugin) (push) Successful in 7m8s
Build Packages / XDS test (neggia plugin) (push) Successful in 7m50s
Build Packages / Generate python client (push) Successful in 16s
Build Packages / Build documentation (push) Successful in 50s
Build Packages / Create release (push) Skipped
v1.0.0-rc.153 (#63)
This is an UNSTABLE release. It includes many experimental features, as well as many AI generated fixes. We recommend using rc.152 for production use.

* jfjoch_broker: Add EXPERIMENTAL pixelrefine mode for image processing
* jfjoch_broker: Allow to load user mask from 8-bit and 16-bit TIFF files
* jfjoch_broker: Add ROI calculation in non-FPGA workflow
* jfjoch_broker: Fixes to TCP image pusher
* jfjoch_broker: Remove NUMA bindings
* jfjoch_broker: Improvements to indexing
* jfjoch_broker: For PSI EIGER, trimming energies are taken from the detector configuration (now compulsory) instead of hardcoded values
* jfjoch_writer: Save ROI definitions and the per-pixel ROI bitmap in the master file; azimuthal ROIs support phi (angular) sectors
* jfjoch_viewer: Major redesign with dockable panels and saved layouts, plus on-canvas creation/move/resize of box, circle and azimuthal ROIs
* jfjoch_viewer: Run jfjoch_process reprocessing jobs from inside the GUI and overlay per-run results

Reviewed-on: #63
2026-06-23 20:29:49 +02:00

174 lines
5.6 KiB
C++

// SPDX-FileCopyrightText: 2026 Filip Leonarski, Paul Scherrer Institute <filip.leonarski@psi.ch>
// SPDX-License-Identifier: GPL-3.0-only
#include "ToolbarIcons.h"
#include <QPainter>
#include <QPainterPath>
#include <QPolygonF>
#include <functional>
namespace {
constexpr int S = 32;
const QColor INK("#2B2B2B");
const QColor DISABLED("#9AA6B3");
using DrawFn = std::function<void(QPainter &, const QColor &)>;
QPixmap render(const DrawFn &fn, const QColor &color) {
QPixmap pm(S, S);
pm.fill(Qt::transparent);
QPainter p(&pm);
p.setRenderHint(QPainter::Antialiasing);
fn(p, color);
return pm;
}
// Toggle-capable icon: ink when off, white when on (checked, navy background), grey when disabled.
QIcon toggleIcon(const DrawFn &fn) {
QIcon ic;
ic.addPixmap(render(fn, INK), QIcon::Normal, QIcon::Off);
ic.addPixmap(render(fn, Qt::white), QIcon::Normal, QIcon::On);
ic.addPixmap(render(fn, Qt::white), QIcon::Active, QIcon::On);
ic.addPixmap(render(fn, DISABLED), QIcon::Disabled, QIcon::Off);
return ic;
}
void fillTriangle(QPainter &p, const QColor &c, QPointF a, QPointF b, QPointF tip) {
p.setPen(Qt::NoPen);
p.setBrush(c);
p.drawPolygon(QPolygonF({a, b, tip}));
}
void bar(QPainter &p, const QColor &c, double x) {
p.setPen(Qt::NoPen);
p.setBrush(c);
p.drawRoundedRect(QRectF(x, 9, 3.2, 14), 1.2, 1.2);
}
// A pair of curved arrows forming a refresh/sync ring.
void syncArrows(QPainter &p, const QColor &c) {
QPen pen(c, 2.4, Qt::SolidLine, Qt::RoundCap);
p.setPen(pen);
p.setBrush(Qt::NoBrush);
const QRectF r(9, 9, 14, 14);
p.drawArc(r, 40 * 16, 230 * 16);
p.drawArc(r, (40 + 180) * 16, 230 * 16);
// arrowheads at the two open ends
p.setPen(Qt::NoPen);
p.setBrush(c);
p.drawPolygon(QPolygonF({QPointF(22, 7.5), QPointF(22, 13.5), QPointF(17.5, 11)}));
p.drawPolygon(QPolygonF({QPointF(10, 24.5), QPointF(10, 18.5), QPointF(14.5, 21)}));
}
}
namespace ToolbarIcons {
QIcon first() {
return toggleIcon([](QPainter &p, const QColor &c) {
bar(p, c, 9);
fillTriangle(p, c, {24, 9}, {24, 23}, {14, 16});
});
}
QIcon prev() {
return toggleIcon([](QPainter &p, const QColor &c) {
fillTriangle(p, c, {22, 8}, {22, 24}, {11, 16});
});
}
QIcon next() {
return toggleIcon([](QPainter &p, const QColor &c) {
fillTriangle(p, c, {10, 8}, {10, 24}, {21, 16});
});
}
QIcon last() {
return toggleIcon([](QPainter &p, const QColor &c) {
fillTriangle(p, c, {8, 9}, {8, 23}, {18, 16});
bar(p, c, 19.8);
});
}
QIcon movie() {
return toggleIcon([](QPainter &p, const QColor &c) {
// Two reels on top of a camera body with a lens barrel — an old-fashioned movie camera.
p.setBrush(Qt::NoBrush);
p.setPen(QPen(c, 1.8));
p.drawEllipse(QPointF(10, 11.5), 3.0, 3.0);
p.drawEllipse(QPointF(16.5, 11.5), 3.0, 3.0);
p.setPen(Qt::NoPen);
p.setBrush(c);
p.drawRoundedRect(QRectF(5, 15, 16, 10), 2, 2);
p.drawRoundedRect(QRectF(21, 17, 5, 6), 1, 1); // lens barrel
});
}
QIcon openFile() {
return toggleIcon([](QPainter &p, const QColor &c) {
// 3.5" diskette ("save" floppy): square jacket with a cut top-right corner, the metal
// shutter across the top, and a label panel below.
p.setPen(QPen(c, 1.8));
p.setBrush(Qt::NoBrush);
QPolygonF body({{6, 6}, {20, 6}, {26, 12}, {26, 26}, {6, 26}});
p.drawPolygon(body); // jacket
p.setPen(Qt::NoPen);
p.setBrush(c);
p.drawRect(QRectF(10, 6, 8, 5.5)); // metal shutter (top)
p.setBrush(Qt::NoBrush);
p.setPen(QPen(c, 1.6));
p.drawRect(QRectF(9.5, 15.5, 13, 8.5)); // label panel
});
}
QIcon reanalyzeImage() {
return toggleIcon([](QPainter &p, const QColor &c) {
p.setPen(QPen(c, 2.0));
p.setBrush(Qt::NoBrush);
p.drawRoundedRect(QRectF(8, 8, 16, 16), 2.5, 2.5);
p.setBrush(c);
p.drawEllipse(QPointF(16, 16), 2.2, 2.2);
});
}
QIcon reanalyzeDataset() {
return toggleIcon([](QPainter &p, const QColor &c) {
p.setPen(QPen(c, 2.0));
p.setBrush(Qt::NoBrush);
for (int i = 2; i >= 0; --i)
p.drawRoundedRect(QRectF(5 + i * 4.0, 5 + i * 4.0, 14, 14), 2.0, 2.0);
p.setBrush(c);
p.drawEllipse(QPointF(12, 12), 2.0, 2.0);
});
}
QIcon httpSync(HttpState state) {
QColor dot;
switch (state) {
case HttpState::Disconnected: dot = QColor("#9AA6B3"); break; // grey
case HttpState::Live: dot = QColor("#5CB85C"); break; // green
case HttpState::Frozen: dot = QColor("#E9A23B"); break; // amber
}
auto draw = [dot](QPainter &p, const QColor &c) {
syncArrows(p, c);
p.setPen(QPen(Qt::white, 1.2));
p.setBrush(dot);
p.drawEllipse(QPointF(24, 24), 4.2, 4.2); // status dot
};
QIcon ic;
ic.addPixmap(render(draw, INK), QIcon::Normal, QIcon::Off);
ic.addPixmap(render(draw, DISABLED), QIcon::Disabled, QIcon::Off);
return ic;
}
QString buttonStyle() {
return QStringLiteral(
"QToolButton { border:none; border-radius:4px; padding:4px 6px; color:#2B2B2B;"
" background:transparent; }"
" QToolButton:hover { background:rgba(250,114,104,0.20); }"
" QToolButton:checked { background:#1F3A5F; color:white; }"
" QToolButton:disabled { color:#9AA6B3; }");
}
} // namespace ToolbarIcons