Display moon-phase for the current date. Complete with navigation for displaying moon-phases before or after the current date. Live version coming soon, which displays the moon-phase as a favicon in your browser's tab.
Created
February 20, 2019 02:38
-
-
Save jesgs/444d4c978c7b6687f09080260e1cd723 to your computer and use it in GitHub Desktop.
Moon Phase
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<div class="wrapper"> | |
<header class="row"> | |
<h1 class="h h1">MoonPhase App</h1> | |
</header> | |
<div class="row"> | |
<ul class="data-table h3"> | |
<li> | |
<strong class="label">Current Date</strong><span class="value js-current-date-value"></span> | |
</li> | |
<li> | |
<strong class="label">Selected Date</strong><span class="js-date-value value"></span> | |
</li> | |
</ul> | |
</div> | |
<div class="row"> | |
<ul class="hlist"> | |
<li> | |
<a href="#" data-name="prev" class="js-date-nav btn">« Back One Day</a> | |
</li> | |
<li> | |
<a href="#" data-name="current" class="js-date-nav btn">Today</a> | |
</li> | |
<li> | |
<a href="#" data-name="next" class="js-date-nav btn">Forward One Day »</a> | |
</li> | |
</ul> | |
</div> | |
<div class="row"> | |
<canvas class="column" id="currentphase" height="288" width="288"> | |
You need a browser that supports HTML5 and JavaScript enabled to see this. | |
</canvas> | |
<div class="column-last"> | |
<h3 class="h h3">Current Phase Data</h3> | |
<ul class="data-table js-moon-data"> | |
<li><strong class="label">Current Phase</strong> <span data-name="currentPhase" class="value"></span></li> | |
<li><strong class="label">Julian Date</strong> <span data-name="julianDate" class="value"></span></li> | |
<li><strong class="label">Moon's Age</strong> <span data-name="synodicAge" class="value"></span></li> | |
<li><strong class="label">Phase Angle</strong> <span data-name="phaseAngle" class="value"></span></li> | |
<li><strong class="label">Percent Illuminated</strong> <span data-name="illumination" class="value"></span></li> | |
<li><strong class="label">Distance</strong> <span data-name="distance" class="value"></span></li> | |
</ul> | |
</div> | |
</div> | |
</div> <!-- .wrapper --> |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
* functions.js | |
*/ | |
function toDegrees(number) { | |
var deg = parseInt(number), | |
min = Math.floor((number - deg) * 60), | |
sec = Math.round((min - parseInt(min)) * 60), | |
degString = deg.toString() + "° " + min.toString() + "′ " + sec.toString() + "″ "; | |
return degString; | |
} | |
function toDaysHours(number) { | |
var day = parseInt(number), | |
hour = ((number - day) * 24).toFixed(0), | |
dayString = day.toString() + "d " + hour + "h"; | |
return dayString; | |
} | |
/* end functions.js */ | |
/* moonfx.js */ | |
MoonFx = (function(){ | |
var MoonFx = function() { | |
/** | |
* Radius of Earth in miles | |
*/ | |
this.EARTH_RADIUS_MI = 3959; | |
/** | |
* Length of one day in seconds | |
*/ | |
this.ONE_DAY = 86400; | |
/** | |
* Moon's synodic period | |
*/ | |
this.SYNODIC_PERIOD = 29.530589; | |
/** | |
* Value of PI in radians | |
*/ | |
this.PI_RADIANS = Math.PI * 2; | |
/** | |
* Current date | |
* | |
* @type long | |
*/ | |
this.moonDate = new Date().getTime(); | |
this.MOON_DATA = { | |
"synodicAge" : toDaysHours(this.getSynodicPhase()), | |
"julianDate" : this.getJulianDate().toFixed(4), | |
"phaseAngle" : toDegrees(this.getPhaseAngle()), | |
"distance" : (this.getDistanceInEarthRadii() * this.EARTH_RADIUS_MI).toFixed(0).toString() + " mi", | |
"illumination" : (this.getIlluminatedRatio() * 100).toFixed(0).toString() + "%" | |
}; | |
return this; | |
}; | |
/** | |
* Set the current date | |
* | |
* @param {long} date | |
* @returns {undefined} | |
*/ | |
MoonFx.prototype.setDate = function(date) { | |
this.moonDate = date; | |
}; | |
/** | |
* Get current date | |
* | |
* @returns {long} | |
*/ | |
MoonFx.prototype.getDate = function() { | |
return this.moonDate; | |
}; | |
/** | |
* Get current Julian Date | |
* | |
* @returns {Number} | |
*/ | |
MoonFx.prototype.getJulianDate = function() { | |
var time = this.getDate(); | |
return ((time / 1000) / this.ONE_DAY) + 2440587.5; | |
}; | |
/** | |
* Get current synodic phase (Moon's age) | |
* | |
* @returns {Number} | |
*/ | |
MoonFx.prototype.getSynodicPhase = function() { | |
var julianDate = this.getJulianDate(), | |
synodicPhase = this._normalize((julianDate - 2451550.1) / this.SYNODIC_PERIOD) | |
* this.SYNODIC_PERIOD; | |
return synodicPhase; | |
}; | |
/** | |
* Get current distance to the moon in Earth Radii | |
* | |
* @returns {Number} | |
*/ | |
MoonFx.prototype.getDistanceInEarthRadii = function() { | |
var distanceInRadians = this._normalize((this.getJulianDate() - 2451562.2) / 27.55454988) * this.PI_RADIANS, | |
synodicPhaseInRadians = this.getSynodicPhase() * this.PI_RADIANS, | |
distance = 60.4 - 3.3 * Math.cos(distanceInRadians) - 0.6 | |
* Math.cos(2 * synodicPhaseInRadians - distanceInRadians) - 0.5 | |
* Math.cos(2 * synodicPhaseInRadians); | |
return distance; | |
}; | |
/** | |
* Get Moon's current ecliptic latitude | |
* @returns {Number} | |
*/ | |
MoonFx.prototype.getEclipticLatitude = function() { | |
var value = this._normalize((this.getJulianDate() - 2451565.2) / 27.212220817), | |
eclipticLatitude = 5.1 * Math.sin(value * this.PI_RADIANS); | |
return eclipticLatitude; | |
}; | |
/** | |
* Get Moon's current ecliptic longitude | |
* @returns {Number|_L1.MoonFx.MoonFx.prototype.getEclipticLongitude.value} | |
*/ | |
MoonFx.prototype.getEclipticLongitude = function() { | |
var synodicPhaseInRadians = this.getSynodicPhase() * this.PI_RADIANS, | |
distanceInRadians = this._normalize((this.getJulianDate() - 245162.2) / 27.55454988) * this.PI_RADIANS, | |
value = this._normalize((this.getJulianDate() - 2451555.8) / 27.321582241), | |
eclipticLongitude = 360 * value + 6.3 + Math.sin(distanceInRadians) + 1.3 | |
* Math.sin(2 * synodicPhaseInRadians - distanceInRadians) + 1.3 | |
* 0.7 * Math.sin(2 * synodicPhaseInRadians); | |
return eclipticLongitude; | |
}; | |
/** | |
* Get the current phase angle | |
* | |
* @param {Number} synodicAge | |
* @returns {Number} | |
*/ | |
MoonFx.prototype.getPhaseAngle = function(synodicAge) { | |
synodicAge = synodicAge ? synodicAge : this.getSynodicPhase(); | |
phaseAngle = synodicAge * (360 / this.SYNODIC_PERIOD); | |
if (phaseAngle > 360) { | |
phaseAngle = phaseAngle - 360; | |
} | |
return phaseAngle; | |
}; | |
/** | |
* Get moon illuminated ratio (in decimals) | |
* @param {Number} synodicAge | |
* @returns {Number} | |
*/ | |
MoonFx.prototype.getIlluminatedRatio = function(synodicAge) { | |
synodicAge = synodicAge ? synodicAge : this.getSynodicPhase(); | |
var phaseAngle = this.getPhaseAngle(synodicAge), | |
ratioOfIllumination = 0.5 * (1 - Math.cos(this._deg2rad(phaseAngle))); | |
return ratioOfIllumination; | |
}; | |
/** | |
* Normalize a number | |
* | |
* @param {Number} value | |
* @returns {Number} | |
*/ | |
MoonFx.prototype._normalize = function(value) { | |
value = value - parseInt(value); | |
if (value < 0){ | |
value = value + 1; | |
} | |
return value; | |
}; | |
/** | |
* Find a number's sign | |
* | |
* @param {Number} $x | |
* @returns {int} | |
*/ | |
MoonFx.prototype._signum = function(x) { | |
return parseInt((Math.abs(x) - x) ? -1 : x > 0); | |
}; | |
/** | |
* Convert degrees to radians | |
* | |
* @param {Number} x | |
* @returns {Number|@exp;Math@pro;PI} | |
*/ | |
MoonFx.prototype._deg2rad = function (x) { | |
return x * (Math.PI / 180); | |
}; | |
return MoonFx; | |
}()); | |
/* end moonfx.js */ | |
/* global.js */ | |
(function($){ | |
// DOM ready | |
$(function(){ | |
MoonPhase.DrawMoon.init(); | |
MoonPhase.Navigation.init(); | |
MoonPhase.DrawFavicon.init(); | |
MoonPhase.LoadData.init(); | |
}); | |
var MoonPhase = MoonPhase || {}; | |
MoonPhase.moonFx = new MoonFx(); | |
MoonPhase.currentTime = new Date().getTime(); | |
MoonPhase.DrawMoon = { | |
canvas : null, | |
context : null, | |
moonFx : null, | |
init: function() { | |
this.canvas = document.getElementById('currentphase'); | |
this.context = this.canvas.getContext('2d'); | |
this.moonFx = MoonPhase.moonFx; | |
this.drawMoon(); | |
}, | |
drawMoon : function() { | |
var ctx = this.context, | |
height = this.canvas.getAttribute('height'), | |
width = this.canvas.getAttribute('width'), | |
cx = width / 2, | |
cy = height / 2, | |
illuminationRatio = this.moonFx.getIlluminatedRatio(), | |
phaseAngle = this.moonFx.getPhaseAngle(); | |
ctx.beginPath(); | |
ctx.arc(cx, cy, (height / 2) - 2, 0, 360, false); | |
ctx.fillStyle = '#fff'; | |
ctx.fill(); | |
ctx.closePath(); | |
// draw limb | |
var points = [[], []]; | |
for (var a = 0; a < 180; a++) { | |
var angle = this.moonFx._deg2rad(a - 90), | |
x1 = Math.ceil( Math.cos( angle ) * cx ), | |
y1 = Math.ceil( Math.sin( angle ) * cy ), | |
moonWidth = x1 * 2, | |
x2 = Math.floor(moonWidth * illuminationRatio); | |
if ( phaseAngle < 180 ) { | |
x1 = cx - x1; | |
x2 = x1 + (moonWidth - x2); | |
} else { // waning | |
x1 = cx + x1; | |
x2 = x1 - (moonWidth - x2); | |
} | |
var y2 = cy + y1, | |
p1 = [x1, y2], | |
p2 = [x2, y2]; | |
points[0].push(p1); | |
points[1].push(p2); | |
} | |
var newPoints = points[0].concat(points[1].reverse()); | |
ctx.beginPath(); | |
ctx.fillStyle = '#000'; | |
for (var n in newPoints) { | |
var p = newPoints[n]; | |
if (n === 0) { | |
ctx.moveTo(p[0], p[1]); | |
} else { | |
ctx.lineTo(p[0], p[1]); | |
} | |
} | |
ctx.fill(); | |
ctx.closePath(); | |
}, | |
changeDate : function(date) { | |
this.moonFx.setDate(date); | |
this.drawMoon(); | |
} | |
}; | |
MoonPhase.LoadData = { | |
init : function() { | |
var moonData = MoonPhase.moonFx.MOON_DATA; | |
$('.js-moon-data .value').each(function(){ | |
var name = $(this).data('name'); | |
$(this).html(moonData[name]); | |
}); | |
} | |
}; | |
MoonPhase.Navigation = { | |
init : function() { | |
$('.js-current-date-value').text(function(){ | |
var dateObj = new Date(); | |
return dateObj.toLocaleDateString() + " @ " + dateObj.toLocaleTimeString(); | |
}); | |
$('.js-date-nav').on('click', function(e){ | |
e.preventDefault(); | |
var action = $(this).data('name'), | |
dateObj = new Date(); | |
if (action === 'prev') { | |
MoonPhase.currentTime = MoonPhase.currentTime - 86400000; | |
} else if (action === 'next') { | |
MoonPhase.currentTime = MoonPhase.currentTime + 86400000; | |
} else { | |
MoonPhase.currentTime = new Date().getTime(); | |
} | |
dateObj.setTime(MoonPhase.currentTime); | |
MoonPhase.moonFx.setDate(MoonPhase.currentTime); | |
$('.js-date-value').text(dateObj.toLocaleDateString() + " @ " + dateObj.toLocaleTimeString()); | |
MoonPhase.DrawMoon.drawMoon(); | |
MoonPhase.DrawFavicon.init(); | |
MoonPhase.LoadData.init(); | |
}).trigger('click'); | |
} | |
}; | |
MoonPhase.DrawFavicon = { | |
init : function() { | |
$('#favicon').remove(); | |
var link = document.createElement('link'), | |
canvas = document.getElementById('currentphase'); | |
link.type = 'image/x-icon'; | |
link.rel = 'shortcut icon'; | |
link.href = canvas.toDataURL("image/x-icon"); | |
link.id = "favicon"; | |
document.getElementsByTagName('head')[0].appendChild(link); | |
} | |
}; | |
})(jQuery); | |
/* end global.js */ |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<script src="https://cdnjs.cloudflare.com/ajax/libs/ocanvas/2.8.3/ocanvas.min.js"></script> | |
<script src="//cdnjs.cloudflare.com/ajax/libs/jquery/2.1.3/jquery.min.js"></script> |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/* | |
Document : screen | |
Created on : Jan 30, 2014, 1:01:26 PM | |
Author : Jess Green <[email protected]> | |
*/ | |
@import url('https://fonts.googleapis.com/css?family=Source+Sans+Pro:200,700,200italic,700italic'); | |
body { | |
font: 1em/1.5 'Source Sans Pro', sans-serif; | |
color: #272822; | |
} | |
canvas { | |
border: 1px solid #000; | |
background-color: #999; | |
} | |
strong { | |
font-weight: 700; | |
} | |
em, cite { | |
font-style: italic; | |
} | |
p { | |
margin-bottom: 1rem; | |
} | |
a, | |
a:link { | |
text-decoration: none; | |
color: #99C62A; | |
} | |
a:visited { | |
color: #66D9EF; | |
} | |
a:hover { | |
color: #F92672; | |
} | |
a:active { | |
color: #FD971F; | |
} | |
.wrapper { | |
width: 960px; | |
margin: 0 auto; | |
} | |
button.btn, | |
input.btn { | |
outline: none; | |
display: inline-block; | |
cursor: pointer; | |
} | |
a.btn { | |
color: #272822; | |
font-size: small; | |
} | |
.btn.btn-narrow { | |
padding: 2px 15px; | |
} | |
.btn { | |
display: inline-block; | |
box-shadow: inset 1px 1px 0 rgba(255, 255, 255, 0.75); | |
background-color: #66D9EF; | |
border-radius: 3px; | |
padding: 5px 15px; | |
border: 1px solid #34A7BD; | |
} | |
.btn:hover { | |
box-shadow: inset -1px -1px 0 rgba(255, 255, 255, 0.75); | |
} | |
/*------------------------------------------------------------------------------ | |
Headers | |
------------------------------------------------------------------------------*/ | |
.h { | |
font-weight: 700; | |
margin-bottom: 1.25rem; | |
display: block; | |
&1 { | |
font-size: 2rem; | |
} | |
&2 { | |
font-size: 1.5rem; | |
} | |
&3 { | |
font-size: 1.25rem; | |
} | |
&4 { | |
font-size: 1rem; | |
} | |
&5 { | |
font-size: 0.875rem; | |
} | |
&6 { | |
font-size: 0.75rem; | |
} | |
} | |
/*------------------------------------------------------------------------------ | |
Horizontal list | |
------------------------------------------------------------------------------*/ | |
.hlist { | |
list-style: none; | |
clear: both; | |
float: none; | |
margin: 0; | |
padding: 0; | |
&:after { | |
display: block; | |
height: 1px; | |
float: none; | |
clear: both; | |
content: "."; | |
visibility: hidden; | |
} | |
> li { | |
float: left; | |
margin-right: 20px; | |
&:last-of-type { | |
margin-right: 0; | |
} | |
} | |
} | |
/*------------------------------------------------------------------------------ | |
Columns/Rows | |
------------------------------------------------------------------------------*/ | |
.row { | |
clear: both; | |
float: none; | |
display: block; | |
margin-bottom: 2.5rem; | |
&:after { | |
display: block; | |
height: 1px; | |
float: none; | |
clear: both; | |
content: "."; | |
visibility: hidden; | |
} | |
} | |
.column { | |
float: left; | |
margin-right: 1.25rem; | |
} | |
.column-last { | |
margin-right: 0; | |
} | |
.small-input { | |
width: 54px; | |
} | |
.moondata { | |
padding-top: 2.5rem; | |
} | |
.data-table { | |
display: table; | |
.label { | |
width: 9.375rem; | |
margin-right: 1.25rem; | |
display: table-cell; | |
&:after { | |
content: ":"; | |
} | |
} | |
.value { | |
display: table-cell; | |
} | |
} | |
.calendar { | |
width: 98%; | |
margin-bottom: 2.5rem; | |
} | |
.calendar th { | |
font-weight: 700; | |
width: 14%; | |
} | |
.calendar td { | |
border: 1px solid; | |
padding: 5px; | |
box-sizing: border-box; | |
text-align: right; | |
height: 135px; | |
} | |
.calendar .moonphase { | |
font-weight: 700; | |
} | |
.hide { | |
display: none; | |
} | |
.show { | |
display: block; | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
where do you find the math for the moon phase ?