summaryrefslogtreecommitdiffstats
path: root/kmoon/kmoonwidget.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kmoon/kmoonwidget.cpp')
-rw-r--r--kmoon/kmoonwidget.cpp369
1 files changed, 369 insertions, 0 deletions
diff --git a/kmoon/kmoonwidget.cpp b/kmoon/kmoonwidget.cpp
new file mode 100644
index 0000000..413b555
--- /dev/null
+++ b/kmoon/kmoonwidget.cpp
@@ -0,0 +1,369 @@
+/*
+ * kmoon - a moon phase indicator
+ * $Id$
+ * Copyright (C) 1998,2000 Stephan Kulow
+ *
+ * This program 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 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#include <stdlib.h>
+#include <assert.h>
+#include <unistd.h>
+
+#include <qbitmap.h>
+#include <qtooltip.h>
+#include <qpainter.h>
+#include <qpopupmenu.h>
+#include <qcolor.h>
+
+#include <dcopclient.h>
+#include <kdebug.h>
+#include <kapplication.h>
+#include <kwin.h>
+#include <kstartupinfo.h>
+#include <kmessagebox.h>
+#include <kaboutdata.h>
+#include <klocale.h>
+#include <kstandarddirs.h>
+#include <kcmdlineargs.h>
+#include <kiconloader.h>
+#include <kiconeffect.h>
+#include <kconfig.h>
+
+#include "kmoondlg.h"
+#include "kmoonwidget.h"
+#include <math.h>
+
+extern double moonphasebylunation(int lun, int phi);
+extern time_t JDtoDate(double jd, struct tm *event_date);
+
+MoonWidget::MoonWidget(QWidget *parent, const char *name)
+ : QWidget(parent, name)
+{
+ struct tm * t;
+ time_t clock;
+
+ counter = -1;
+ KConfig *config = KGlobal::config();
+ config->setGroup("General");
+ _angle = config->readNumEntry("Rotation", 0);
+ _north = config->readBoolEntry("Northern", true);
+ _mask = config->readBoolEntry("Mask", true);
+ old_angle = old_w = old_h = old_counter = -1;
+ old_north = false;
+ old_mask = false;
+ startTimer(1000 * 60 * 20);
+
+ time(&clock);
+ t = gmtime(&clock);
+ // kdDebug() << "time " << t->tm_isdst << " " << timezone << " " << daylight << endl ;
+ calcStatus(mktime(t));
+}
+
+MoonWidget::~MoonWidget()
+{
+}
+
+void MoonWidget::calcStatus( time_t time )
+{
+ uint lun = 0;
+ time_t last_new = 0;
+ time_t next_new = 0;
+
+ do {
+ double JDE = moonphasebylunation(lun, 0);
+ last_new = next_new;
+ next_new = JDtoDate(JDE, 0);
+ lun++;
+ } while (next_new < time);
+
+ lun -= 2;
+
+ QDateTime ln;
+ ln.setTime_t( last_new );
+ kdDebug() << KGlobal::locale()->formatDateTime( ln ) << endl;
+
+ time_t first_quarter = JDtoDate( moonphasebylunation( lun, 1 ), 0 );
+ QDateTime fq;
+ fq.setTime_t( first_quarter );
+ kdDebug() << KGlobal::locale()->formatDateTime( fq ) << endl;
+
+ time_t full_moon = JDtoDate( moonphasebylunation( lun, 2 ), 0 );
+ QDateTime fm;
+ fm.setTime_t( full_moon );
+ kdDebug() << KGlobal::locale()->formatDateTime( fm ) << endl;
+
+ time_t third_quarter = JDtoDate( moonphasebylunation( lun, 3 ), 0 );
+ QDateTime tq;
+ tq.setTime_t( third_quarter );
+ kdDebug() << KGlobal::locale()->formatDateTime( tq ) << endl;
+
+ QDateTime nn;
+ nn.setTime_t( next_new );
+ kdDebug() << KGlobal::locale()->formatDateTime( nn ) << endl;
+
+ QDateTime now;
+ now.setTime_t( time );
+ kdDebug() << KGlobal::locale()->formatDateTime( now ) << endl;
+
+ counter = ln.daysTo( now );
+ kdDebug() << "counter " << counter << " " << fm.daysTo( now ) << endl;
+
+ if ( fm.daysTo( now ) == 0 ) {
+ counter = 14;
+ tooltip = i18n( "Full Moon" );
+ return;
+ } else if ( counter <= 15 && counter >= 13 ) {
+ counter = 14 + fm.daysTo( now );
+ kdDebug() << "around full moon " << counter << endl;
+ }
+
+ int diff = fq.daysTo( now );
+ if ( diff == 0 )
+ counter = 7;
+ else if ( counter <= 8 && counter >= 6 ) {
+ counter = 7 + diff;
+ kdDebug() << "around first quarter " << counter << endl;
+ }
+
+ diff = ln.daysTo( now );
+ if ( diff == 0 )
+ counter = 0;
+ else if ( counter <= 1 || counter >= 28 )
+ {
+ counter = ( 29 + diff ) % 29;
+ diff = -nn.daysTo( now );
+ if ( diff == 0 )
+ counter = 0;
+ else if ( diff < 3 )
+ counter = 29 - diff;
+ kdDebug() << "around new " << counter << " " << diff << endl;
+ }
+
+ if ( tq.daysTo( now ) == 0 )
+ counter = 21;
+ else if ( counter <= 22 && counter >= 20 )
+ {
+ counter = 21 + tq.daysTo( now );
+ kdDebug() << "around third quarter " << counter << endl;
+ }
+
+ kdDebug() << "counter " << counter << endl;
+
+ assert (counter >= 0 && counter < 29);
+
+ switch (counter) {
+ case 0:
+ tooltip = i18n("New Moon");
+ return;
+ case 1:
+ case 2:
+ case 3:
+ case 4:
+ case 5:
+ case 6:
+ tooltip = i18n("Waxing Crescent (New Moon was yesterday)", "Waxing Crescent (%n days since New Moon)", counter );
+ break;
+ case 7:
+ tooltip = i18n("First Quarter");
+ break;
+ case 8:
+ case 9:
+ case 10:
+ case 11:
+ case 12:
+ case 13:
+ tooltip = i18n( "Waxing Gibbous (Tomorrow is Full Moon)", "Waxing Gibbous (%n days to Full Moon)", -fm.daysTo( now ) );
+ break;
+ case 14:
+ assert( false );
+ break;
+ case 15:
+ case 16:
+ case 17:
+ case 18:
+ case 19:
+ case 20:
+ tooltip = i18n("Waning Gibbous (Yesterday was Full Moon)", "Waning Gibbous (%n days since Full Moon)", fm.daysTo( now ) );
+ break;
+ case 21:
+ tooltip = i18n("Last Quarter");
+ break;
+ case 22:
+ case 23:
+ case 24:
+ case 25:
+ case 26:
+ case 27:
+ case 28:
+ kdDebug() << "nn.days " << ln.daysTo( now ) << " " << nn.daysTo( now ) << endl;
+ tooltip = i18n("Waning Crescent (Tomorrow is New Moon)", "Waning Crescent (%n days to New Moon)", -nn.daysTo( now ) );
+ break;
+ default:
+ kdFatal() << "coolo can't count\n";
+ }
+
+ renderGraphic();
+ repaint();
+}
+
+QImage MoonWidget::loadMoon(int index)
+{
+ if (index == 0) // the new moon has the wrong filename
+ index = 29;
+ QString filename = QString("kmoon/pics/moon%1.png").arg(index);
+ QString path = locate("data", filename);
+ if (path.isNull())
+ kdFatal() << "cound't find " << filename << ". Exiting.\n";
+ QImage image(path);
+ KIconEffect iconeffect;
+ image=iconeffect.apply(image, KIcon::Panel, KIcon::DefaultState);
+ return image;
+}
+
+void MoonWidget::setAngle(int value)
+{
+ _angle = value;
+ renderGraphic();
+ repaint();
+}
+
+void MoonWidget::setNorthHemi(bool n)
+{
+ _north = n;
+ renderGraphic();
+ repaint();
+}
+
+void MoonWidget::setMask(bool value)
+{
+ _mask = value;
+ renderGraphic();
+ repaint();
+}
+
+void MoonWidget::paintEvent(QPaintEvent *)
+{
+ bitBlt(this, 0, 0, &pixmap, 0, 0);
+}
+
+void MoonWidget::resizeEvent(QResizeEvent *)
+{
+ renderGraphic();
+ repaint();
+}
+
+void MoonWidget::renderGraphic()
+{
+ if (old_counter == counter && old_w == width() && old_h == height() &&
+ old_angle == _angle && old_north == _north)
+ return;
+ old_counter = counter;
+ old_w = width();
+ old_h = height();
+ old_north = _north;
+ QImage im = loadMoon(counter);
+ assert(!im.isNull());
+ im = im.convertDepth(32, 0);
+ assert(!im.isNull());
+
+ int mw = QMIN(width(), height());
+ QImage dest;
+
+ if (QPixmap::defaultDepth() > 8) {
+
+ if (fabs(_angle)!=0) { // nothing to rotate
+ //We expand the image 2x before rotating, rotate it, and then average out
+ //the pixel for better quality
+ int dmw = 2 * mw;
+ if (!pixmap.convertFromImage(im.smoothScale(dmw, dmw), 0)) {
+ return;
+ }
+ QWMatrix m;
+ m.rotate(_angle);
+ QPixmap rotated = pixmap.xForm(m);
+
+ //Copy the relevant part back to the pixmap
+ QRegion r(QRect(0, 0, dmw, dmw), QRegion::Ellipse);
+ QPainter p;
+ p.begin(&pixmap);
+ p.fillRect(0, 0, dmw, dmw, Qt::black);
+ p.setClipRegion(r);
+ p.drawPixmap(0, 0, rotated, (rotated.width() - dmw) / 2,
+ (rotated.height() - dmw) / 2,
+ dmw, dmw);
+ p.end();
+
+ //Shrink down to the proper size, averaging from the source
+ im = pixmap.convertToImage();
+ dest = im.copy(0, 0, mw, mw);
+ for (int y = 0; y < mw; y++) {
+ QRgb *destline = (QRgb*)dest.scanLine(y);
+ QRgb *sourceline1 = (QRgb*)im.scanLine(2*y);
+ QRgb *sourceline2 = (QRgb*)im.scanLine(2*y + 1);
+ for (int x = 0; x < mw; x++) {
+ int r = qRed(sourceline1[2*x]) + qRed(sourceline1[2*x+1]);
+ r = r + qRed(sourceline2[2*x]) + qRed(sourceline2[2*x+1]);
+ int g = qGreen(sourceline1[2*x]) + qGreen(sourceline1[2*x+1]);
+ g = g + qGreen(sourceline2[2*x]) + qGreen(sourceline2[2*x+1]);
+ int b = qBlue(sourceline1[2*x]) + qBlue(sourceline1[2*x+1]);
+ b = b + qBlue(sourceline2[2*x]) + qBlue(sourceline2[2*x+1]);
+ destline[x] = qRgb(qRound(r / 4), qRound(g / 4),
+ qRound(b / 4));
+ }
+ }
+ } else {
+ dest = im.smoothScale(mw, mw).convertDepth(32);
+ }
+ if (_mask) {
+ // generate alpha-channel
+ int dmw = mw*2;
+ QBitmap dMask(dmw, dmw);
+ QRegion r(QRect(0, 0, dmw, dmw), QRegion::Ellipse);
+ QPainter p;
+ p.begin(&dMask);
+ p.fillRect(0, 0, dmw, dmw, Qt::white);
+ p.setClipRegion(r);
+ p.fillRect(0, 0, dmw, dmw, Qt::black);
+ p.end();
+ QImage Mask2 = dMask.convertToImage().convertDepth(32).smoothScale(mw, mw);
+ dest.setAlphaBuffer(true);
+ for (int y = 0; y < mw; y++) {
+ QRgb *destline = (QRgb*)dest.scanLine(y);
+ QRgb *sourceline = (QRgb*)Mask2.scanLine(y);
+ for (int x = 0; x < mw; x++) {
+ destline[x] = (destline[x] & RGB_MASK)|(qRed(sourceline[x]) << 24);
+ }
+ }
+ }
+ } else {
+ dest = im.smoothScale(mw, mw);
+ }
+
+ if (!_north)
+ dest = dest.mirror(true, true);
+
+ if (!pixmap.convertFromImage(dest, 0)) {
+ return;
+ }
+
+ QToolTip::remove(this);
+
+ QToolTip::add(this, tooltip);
+}
+
+
+#include "kmoonwidget.moc"