Kstars

avtplotwidget.cpp
1/*
2 SPDX-FileCopyrightText: 2007 Jason Harris <kstars@30doradus.org>
3
4 SPDX-License-Identifier: GPL-2.0-or-later
5*/
6
7#include "avtplotwidget.h"
8
9#include "kstarsdata.h"
10#include "Options.h"
11
12#include <QWidget>
13#include <QDebug>
14#include <QLinearGradient>
15#include <QMouseEvent>
16#include <QPainter>
17#include <QTime>
18#include <QToolTip>
19
20#include <KLocalizedString>
21#include <kplotobject.h>
22#include <kplotpoint.h>
23
24#include "kplotaxis.h"
25#include "ksalmanac.h"
26
27AVTPlotWidget::AVTPlotWidget(QWidget *parent) : KPlotWidget(parent)
28{
29 setAntialiasing(true);
30
31 MousePoint = QPoint(-1, -1);
32}
33
38
40{
41 Q_UNUSED(e);
42 MousePoint = QPoint(-1, -1);
43 update();
44}
45
47{
48 QRect checkRect(leftPadding(), topPadding(), pixRect().width(), pixRect().height());
49 int Xcursor = e->x();
50 int Ycursor = e->y();
51
52 if (!checkRect.contains(e->x(), e->y()))
53 {
54 if (e->x() < checkRect.left())
55 Xcursor = checkRect.left();
56 if (e->x() > checkRect.right())
57 Xcursor = checkRect.right();
58 if (e->y() < checkRect.top())
59 Ycursor = checkRect.top();
60 if (e->y() > checkRect.bottom())
61 Ycursor = checkRect.bottom();
62 }
63
64 Xcursor -= leftPadding();
65 Ycursor -= topPadding();
66
67 MousePoint = QPoint(Xcursor, Ycursor);
68 displayToolTip(e->pos(), e->globalPos());
69 update();
70}
71
73{
74 if (event->type() == QEvent::ToolTip)
75 {
76 QHelpEvent *helpEvent = static_cast<QHelpEvent *>(event);
77 displayToolTip(helpEvent->pos(), helpEvent->globalPos());
78 return true;
79 }
80 return QWidget::event(event);
81}
82
83// Map logical data coordinates to mouse coordinates.
84QPointF AVTPlotWidget::toXY(double vx, double vy)
85{
86 QRectF plotArea = pixRect();
87 double px = leftPadding() + ((vx - xMin) * plotArea.width()) / (xMax - xMin);
88 // Top of the plot is y=0
89 double py = topPadding() + ((altitudeAxisMax - vy) * plotArea.height()) / (altitudeAxisMax - altitudeAxisMin);
90 return QPointF(px, py);
91}
92
93void AVTPlotWidget::displayToolTip(const QPoint &pos, const QPoint &globalPos)
94{
95 for (const auto &tip : tips)
96 {
97 for (const auto &pt : tip.points)
98 {
99 if (qAbs(pt.x() - pos.x()) < 5 && qAbs(pt.y() - pos.y()) < 5)
100 {
101 QToolTip::showText(globalPos, tip.label, this, QRect(), 3000);
102 return;
103 }
104 }
105 }
107}
108
109
110// All the int coordinates (rise, set) need to be converted from hours relative to midnight
111// into graph coordinates before calling this.
112void drawMoon(QPainter &p, int rise, int set, int fade, const QColor &color, int width, int height, double leftPadding)
113{
114 QBrush brush(color, Qt::Dense5Pattern);
115 QBrush dimmerBrush(color, Qt::Dense6Pattern);
116 QBrush dimmestBrush(color, Qt::Dense7Pattern);
117 QRectF r;
118 if (set < rise)
119 {
120 if (set + fade >= leftPadding && set - fade < leftPadding + width)
121 {
122 r = QRectF(leftPadding, 0.0, (set - fade) - leftPadding, height);
123 p.fillRect(r, brush);
124 r = QRectF(set - fade, 0.0, fade, height);
125 p.fillRect(r, dimmerBrush);
126 r = QRectF(set, 0.0, fade, height);
127 p.fillRect(r, dimmestBrush);
128 }
129 if (rise + fade >= leftPadding && rise - fade < leftPadding + width)
130 {
131 r = QRectF(rise - fade, 0.0, fade, height);
132 p.fillRect(r, dimmestBrush);
133 r = QRectF(rise, 0.0, fade, height);
134 p.fillRect(r, dimmerBrush);
135
136 // Since set < rise, we draw to the end of the box
137 r = QRectF(rise + fade, 0.0, width, height);
138 p.fillRect(r, brush);
139 }
140 }
141 else
142 {
143 r = QRectF(rise - fade, 0.0, fade, height);
144 p.fillRect(r, dimmestBrush);
145 r = QRectF(rise, 0.0, fade, height);
146 p.fillRect(r, dimmerBrush);
147 r = QRectF(rise + fade, 0.0, (set - rise) - 2 * fade, height);
148 p.fillRect(r, brush);
149 r = QRectF(set - fade, 0.0, fade, height);
150 p.fillRect(r, dimmerBrush);
151 r = QRectF(set, 0.0, fade, height);
152 p.fillRect(r, dimmestBrush);
153 }
154}
155
156// All the int coordinates (rise, set, da, du) need to be converted from hours relative to midnight
157// into graph coordinates before calling this.
158void drawSun(QPainter &p, int rise, int set, double minAlt, double maxAlt, int da, int du, bool noDawn,
159 const QColor &color, int width, int height)
160{
161 if (maxAlt < 0.0 && minAlt < -18.0)
162 {
163 // The sun never rise but the sky is not completely dark
164 QLinearGradient grad = QLinearGradient(QPointF(0.0, 0.0), QPointF(du, 0.0));
165
166 QColor gradStartColor = color;
167 gradStartColor.setAlpha((1 - (maxAlt / -18.0)) * 255);
168
169 grad.setColorAt(0, gradStartColor);
171 p.fillRect(QRectF(0.0, 0.0, du, height), grad);
172 grad.setStart(QPointF(width, 0.0));
173 grad.setFinalStop(QPointF(da, 0.0));
174 p.fillRect(QRectF(da, 0.0, width, height), grad);
175 }
176 else if (maxAlt < 0.0 && minAlt > -18.0)
177 {
178 // The sun never rise but the sky is NEVER completely dark
179 QLinearGradient grad = QLinearGradient(QPointF(0.0, 0.0), QPointF(width, 0.0));
180
181 QColor gradStartEndColor = color;
182 gradStartEndColor.setAlpha((1 - (maxAlt / -18.0)) * 255);
183 QColor gradMidColor = color;
184 gradMidColor.setAlpha((1 - (minAlt / -18.0)) * 255);
185
186 grad.setColorAt(0, gradStartEndColor);
187 grad.setColorAt(0.5, gradMidColor);
188 grad.setColorAt(1, gradStartEndColor);
189 p.fillRect(QRectF(0.0, 0.0, width, height), grad);
190 }
191 else if (noDawn)
192 {
193 // The sun sets and rises but the sky is never completely dark
194 p.fillRect(0, 0, set, int(0.5 * height), color);
195 p.fillRect(rise, 0, width, int(0.5 * height), color);
196
197 QLinearGradient grad = QLinearGradient(QPointF(set, 0.0), QPointF(rise, 0.0));
198
199 QColor gradMidColor = color;
200 gradMidColor.setAlpha((1 - (minAlt / -18.0)) * 255);
201
202 grad.setColorAt(0, color);
203 grad.setColorAt(0.5, gradMidColor);
204 grad.setColorAt(1, color);
205 p.fillRect(QRectF(set, 0.0, rise - set, height), grad);
206 }
207 else
208 {
209 if (set > 0)
210 p.fillRect(0, 0, set, height, color);
211 if (rise < width)
212 p.fillRect(rise, 0, width, height, color);
213
214 QLinearGradient grad = QLinearGradient(QPointF(set, 0.0), QPointF(du, 0.0));
215 grad.setColorAt(0, color);
217 p.fillRect(QRectF(set, 0.0, du - set, height), grad);
218
219 grad.setStart(QPointF(rise, 0.0));
220 grad.setFinalStop(QPointF(da, 0.0));
221 p.fillRect(QRectF(da, 0.0, rise - da, height), grad);
222 }
223}
224
225// This legacy code always plotted from noon to noon (24 hours starting at noon).
226// To generalize this code, we still compute noon-to-noon coords, but then convert
227// them to more general plot coordinates where the plot length isn't 24 hours and
228// the plot doesn't begin at noon.
229int AVTPlotWidget::convertCoords(double xCoord)
230{
231 const double plotWidth = pixRect().width();
232 const double pixelsPerHour = plotWidth / plotDuration;
233 const double newPosition = pixelsPerHour * ((xCoord * 24.0 / plotWidth) - noonOffset);
234 return newPosition;
235}
236
237namespace
238{
239double findYValue(const KPlotObject *po, double x)
240{
241 const auto points = po->points();
242 const int size = points.size();
243 if (size == 0) return 0;
244 if (x < points[0]->x()) return points[0]->y();
245 if (x > points[size - 1]->x()) return points[size - 1]->y();
246 for (int i = 0; i < size - 1; ++i)
247 {
248 const double ix = points[i]->x();
249 const double iy = points[i]->y();
250 const double nextIx = points[i + 1]->x();
251 const double nextIy = points[i + 1]->y();
252 if (x == ix) return iy;
253 if (x == nextIx) return nextIy;
254 if (x > ix && x < nextIx)
255 return iy + (nextIy - iy) * (x - ix) / (nextIx - ix);
256 }
257 return points[size - 1]->y();
258}
259} // namespace
260
262{
263 Q_UNUSED(e)
264
265 QPainter p;
266
267 p.begin(this);
271
272 setPixRect();
273 p.setClipRect(pixRect());
274 p.setClipping(true);
275
276 int pW = pixRect().width();
277 int pH = pixRect().height();
278
279 QColor SkyColor(0, 100, 200);
280 /*
281 if (Options::darkAppColors())
282 SkyColor = QColor(200, 0, 0); // use something red, visible through a red filter
283 */
284
285 // Draw gradient representing lunar interference in the sky
286 if (MoonIllum > 0.01) // do this only if Moon illumination is reasonable so it's important
287 {
288 double moonrise = pW * (0.5 + MoonRise);
289 double moonset = pW * (MoonSet - 0.5);
290 if (moonset < 0)
291 moonset += pW;
292 if (moonrise > pW)
293 moonrise -= pW;
294 moonrise = convertCoords(moonrise);
295 moonset = convertCoords(moonset);
296
297 if (moonrise > pW)
298 {
299 const double pixelsPerHour = pW * 1.0 / plotDuration;
300 moonrise -= 24 * pixelsPerHour;
301 }
302 const int mooncolor = int(10 + MoonIllum * 130);
303 const QColor MoonColor(mooncolor, mooncolor, mooncolor);
304 int fadewidth =
305 pW *
306 0.01; // pW * fraction of day to fade the moon brightness over (0.01 corresponds to roughly 15 minutes, 0.007 to 10 minutes), both before and after actual set.
307
308 drawMoon(p, int(moonrise), int(moonset), fadewidth, MoonColor, pW, pH, leftPadding());
309
310 }
311 //draw daytime sky if the Sun rises for the current date/location
312 if (SunMaxAlt > -18.0)
313 {
314 // Initially compute centered on midnight, so modulate dawn/dusk by 0.5
315 // Then convert to general coordinates.
316 int rise = convertCoords(pW * (0.5 + SunRise));
317 int set = convertCoords(pW * (SunSet - 0.5));
318 int dawn = convertCoords(pW * (0.5 + Dawn));
319 double dusk = int(pW * (Dusk - 0.5));
320 if (dusk < 0) dusk = pW + dusk;
321 dusk = convertCoords(dusk);
322
323 if (SunMinAlt > 0.0)
324 {
325 // The sun never set and the sky is always blue
326 p.fillRect(rect(), SkyColor);
327 }
328 else drawSun(p, rise, set, SunMinAlt, SunMaxAlt, dawn, int(dusk), Dawn < 0.0, SkyColor, pW, pH);
329 }
330
331 //draw ground
332 if (altitudeAxisMin < 0)
333 {
334 const int groundYValue = pH + altitudeAxisMin * pH / (altitudeAxisMax - altitudeAxisMin);
335 p.fillRect(0, groundYValue, pW, groundYValue,
336 KStarsData::Instance()->colorScheme()->colorNamed(
337 "HorzColor")); // asimha changed to use color from scheme. Formerly was QColor( "#002200" )
338 }
339
340 foreach (KPlotObject *po, plotObjects())
341 {
342 po->draw(&p, this);
343 }
344
345 p.setClipping(false);
346 drawAxes(&p);
347
348 //Add vertical line indicating "now"
349 QFont smallFont = p.font();
350 smallFont.setPointSize(smallFont.pointSize()); // wat?
351 if (geo)
352 {
354 .time(); // convert the current system clock time to the TZ corresponding to geo
355 double x = 12.0 + t.hour() + t.minute() / 60.0 + t.second() / 3600.0;
356 while (x > 24.0)
357 x -= 24.0;
358 double ix = x * pW / 24.0; //convert to screen pixel coords
359 ix = convertCoords(ix);
360 p.setPen(QPen(QBrush("white"), 2.0, Qt::DotLine));
361 p.drawLine(int(ix), 0, ix, pH);
362
363 //Label this vertical line with the current time
364 p.save();
365 p.setFont(smallFont);
366 p.translate(int(ix) + 10, pH - 20);
367 p.rotate(-90);
368 p.drawText(
369 0, 0,
370 QLocale().toString(t, QLocale::ShortFormat)); // short format necessary to avoid false time-zone labeling
371 p.restore();
372 }
373
374 //Draw crosshairs at clicked position
375 if (MousePoint.x() > 0)
376 {
377 p.setPen(QPen(QBrush("gold"), 1.0, Qt::SolidLine));
378 p.drawLine(QLineF(MousePoint.x() + 0.5, 0.5, MousePoint.x() + 0.5, pixRect().height() - 0.5));
379
380 //Label each crosshair line (time and altitude)
381 p.setFont(smallFont);
382
383 double h = (MousePoint.x() * plotDuration) / pW - (12.0 - noonOffset);
384 double a = 0;
385 if (currentLine >= 0 && currentLine < plotObjects().size())
386 a = findYValue(plotObjects()[currentLine], h);
387 p.drawText(15, 15, QString::number(a, 'f', 1) + QChar(176));
388
389 if (h < 0.0)
390 h += 24.0;
391 QTime t = QTime(int(h), int(60. * (h - int(h))));
392 p.save();
393 p.translate(MousePoint.x() + 10, pH - 20);
394 p.rotate(-90);
395 p.drawText(
396 0, 0,
397 QLocale().toString(t, QLocale::ShortFormat)); // short format necessary to avoid false time-zone labeling
398 p.restore();
399 }
400
401 p.end();
402}
403
405{
406 if (index >= 0 && index < plotObjects().size())
407 currentLine = index;
408}
409
410void AVTPlotWidget::setDawnDuskTimes(double da, double du)
411{
412 Dawn = da;
413 Dusk = du;
414 update(); // fixme: should we always be calling update? It's probably cheap enough that we can.
415}
416
417void AVTPlotWidget::setMinMaxSunAlt(double min, double max)
418{
419 SunMinAlt = min;
420 SunMaxAlt = max;
421 update();
422}
423
424void AVTPlotWidget::setSunRiseSetTimes(double sr, double ss)
425{
426 SunRise = sr;
427 SunSet = ss;
428 update();
429}
430
431void AVTPlotWidget::setMoonRiseSetTimes(double mr, double ms)
432{
433 MoonRise = mr;
434 MoonSet = ms;
435 update();
436}
437
439{
440 MoonIllum = mi;
441 update();
442}
443
444void AVTPlotWidget::setPlotExtent(double offset, double duration)
445{
446 noonOffset = offset;
447 plotDuration = duration;
448}
449
450void AVTPlotWidget::disableAxis(KPlotWidget::Axis axisToDisable)
451{
452 axis(axisToDisable)->setVisible(false);
453}
454
455void AVTPlotWidget::plot(const GeoLocation *geo, KSAlmanac *ksal, const QVector<double> &times,
456 const QVector<double> &alts, int lineWidth, Qt::GlobalColor color, const QString &label)
457{
458 currentLine = 0;
459 xMin = times[0];
460 xMax = times.last();
461 setLimits(xMin, xMax, altitudeAxisMin, altitudeAxisMax);
462
463 setSecondaryLimits(times[0], times.last(), altitudeAxisMin, altitudeAxisMax);
467 setGeoLocation(geo);
468
469 setSunRiseSetTimes(ksal->getSunRise(), ksal->getSunSet());
470 setDawnDuskTimes(ksal->getDawnAstronomicalTwilight(), ksal->getDuskAstronomicalTwilight());
471 setMinMaxSunAlt(ksal->getSunMinAlt(), ksal->getSunMaxAlt());
472 setMoonRiseSetTimes(ksal->getMoonRise(), ksal->getMoonSet());
473 setMoonIllum(ksal->getMoonIllum());
474
475 const double noonOffset = times[0] - -12;
476 const double plotDuration = times.last() - times[0];
477 setPlotExtent(noonOffset, plotDuration);
479
480 tips.clear();
481 plotOverlay(times, alts, lineWidth, color, label);
482}
483
484void AVTPlotWidget::plotOverlay(const QVector<double> &times, const QVector<double> &alts, int lineWidth,
485 Qt::GlobalColor color, const QString &label)
486{
488 QPen pen;
489 pen.setWidth(lineWidth);
490 pen.setColor(color);
491 po->setLinePen(pen);
492
493 Tip tip;
494 tip.label = label;
495 for (int i = 0; i < times.size(); ++i)
496 {
497 po->addPoint(times[i], alts[i]);
498 if (!label.isEmpty())
499 {
500 QPointF p = toXY(times[i], alts[i]);
501 tip.points.append(p);
502 }
503 }
504 addPlotObject(po);
505 if (!label.isEmpty())
506 tips.append(tip);
507 update();
508}
509
510void AVTPlotWidget::setAltitudeAxis(double min, double max)
511{
512 if (min < max)
513 {
514 altitudeAxisMin = min;
515 altitudeAxisMax = max;
516 }
517}
518
void setGeoLocation(const GeoLocation *geo_)
Set the GeoLocation.
void setSunRiseSetTimes(double sr, double ss)
Set the fractional positions of the Sunrise and Sunset positions, in units where last midnight was 0....
bool event(QEvent *event) override
Used to catch tooltip event, otherwise calls parent.
void setMoonRiseSetTimes(double mr, double ms)
Set the fractional positions of moonrise and moon set in units where last midnight was 0....
void paintEvent(QPaintEvent *e) override
Redraw the plot.
void setPlotExtent(double noonOffset, double plotDuration)
This is needed when not plotting from noon to noon.
void setCurrentLine(int lineIndex)
Sets the plot index which whose altitude is displayed when clicking on the graph.
void plot(const GeoLocation *geo, KSAlmanac *ksal, const QVector< double > &times, const QVector< double > &alts, int lineWidth=2, Qt::GlobalColor color=Qt::white, const QString &label="")
Higher level method to plot.
void mouseDoubleClickEvent(QMouseEvent *e) override
Reset the MousePoint to a null value, to erase the crosshairs.
void setMoonIllum(double mi)
Set the moon illumination.
void mousePressEvent(QMouseEvent *e) override
Simply calls mouseMoveEvent().
void mouseMoveEvent(QMouseEvent *e) override
Handle mouse move events.
void setAltitudeAxis(double min, double max)
Sets the Y-axis min and max values.
Contains all relevant information for specifying a location on Earth: City Name, State/Province name,...
Definition geolocation.h:28
void setTickLabelFormat(char format='g', int fieldWidth=0, int precision=-1)
void setTickLabelsShown(bool b)
void setVisible(bool visible)
void draw(QPainter *p, KPlotWidget *pw)
void addPoint(const QPointF &p, const QString &label=QString(), double barWidth=0.0)
void setLinePen(const QPen &p)
QList< KPlotPoint * > points() const
QList< KPlotObject * > plotObjects() const
void setPixRect()
void setSecondaryLimits(double x1, double x2, double y1, double y2)
int leftPadding() const
int topPadding() const
KPlotAxis * axis(Axis type)
virtual void drawAxes(QPainter *p)
void setLimits(double x1, double x2, double y1, double y2)
void addPlotObject(KPlotObject *object)
QRect pixRect() const
QColor backgroundColor() const
bool antialiasing() const
void removeAllPlotObjects()
A class that implements methods to find sun rise, sun set, twilight begin / end times,...
Definition ksalmanac.h:27
double getSunMaxAlt() const
These functions return the max and min altitude of the sun during the course of the day in degrees.
Definition ksalmanac.h:75
double getMoonIllum() const
Definition ksalmanac.h:86
double getSunRise() const
All the functions returns the fraction of the day given by getDate() as their return value.
Definition ksalmanac.h:65
static KStarsDateTime currentDateTimeUtc()
QString label(StandardShortcut id)
void setAlpha(int alpha)
int pointSize() const const
void setPointSize(int pointSize)
void setColorAt(qreal position, const QColor &color)
const QPoint & globalPos() const const
const QPoint & pos() const const
void setFinalStop(const QPointF &stop)
void setStart(const QPointF &start)
T & last()
qsizetype size() const const
QPoint globalPos() const const
QPoint pos() const const
int x() const const
int y() const const
bool begin(QPaintDevice *device)
void drawLine(const QLine &line)
void drawText(const QPoint &position, const QString &text)
bool end()
void fillRect(const QRect &rectangle, QGradient::Preset preset)
const QFont & font() const const
void restore()
void rotate(qreal angle)
void save()
void setClipRect(const QRect &rectangle, Qt::ClipOperation operation)
void setClipping(bool enable)
void setFont(const QFont &font)
void setPen(Qt::PenStyle style)
void setRenderHint(RenderHint hint, bool on)
void translate(const QPoint &offset)
void setColor(const QColor &color)
void setWidth(int width)
int bottom() const const
bool contains(const QPoint &point, bool proper) const const
int height() const const
int left() const const
int right() const const
int top() const const
int width() const const
qreal height() const const
qreal width() const const
bool isEmpty() const const
QString number(double n, char format, int precision)
Dense5Pattern
transparent
int hour() const const
int minute() const const
int second() const const
void hideText()
void showText(const QPoint &pos, const QString &text, QWidget *w, const QRect &rect, int msecDisplayTime)
virtual bool event(QEvent *event) override
void update()
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Mar 28 2025 11:57:25 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.