Skip to content

Commit

Permalink
feat: add findConjunctions() utility to conjunction module in @observ…
Browse files Browse the repository at this point in the history
…erly/astrometry

feat: add findConjunctions() utility to conjunction module in @observerly/astrometry
  • Loading branch information
michealroberts committed Nov 10, 2024
1 parent fcb67ea commit 0839756
Showing 1 changed file with 137 additions and 0 deletions.
137 changes: 137 additions & 0 deletions src/conjunction.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,8 @@ import type {

import { convertEclipticToEquatorial, convertEquatorialToHorizontal } from './coordinates'

import { getLunarEquatorialCoordinate } from './moon'

import {
getPlanetaryGeocentricEclipticCoordinate,
getPlanetaryPositions,
Expand Down Expand Up @@ -394,3 +396,138 @@ export const findPlanetaryConjunctions = (
}

/*****************************************************************************************************************/

/**
* findConjunctions
*
* Finds all conjunctions of the planets, Spica, Regulus and the Moon within a given time interval,
* returning only those that are inconjunction with each other (as determined by the angular
* separation threshold).
*
* @param interval - The interval to search for the initial conjunction.
* @param observer - The geographic coordinate of the observer.
* @param horizon - The minimum altitude of the planets above the horizon.
* @param angularSeparationThreshold - The minimum angular separation for conjunction.
* @param stepMinutes - The step size in minutes for checking conjunction.
* @returns An array of conjunctions found.
*
*/
export const findConjunctions = (
interval: Interval,
observer: GeographicCoordinate,
params: {
horizon?: number // six degrees above the horizon
angularSeparationThreshold?: number // three degrees of separation
stepMinutes?: number // check every 1/3 hour
} = {
horizon: 6,
angularSeparationThreshold: ANGULAR_SEPARATION_THRESHOLD,
stepMinutes: 20
}
): Map<string, Conjunction> => {
// A conjunction is a close apparent approach of two celestial objects in the sky.
const conjunctions = new Map<string, Conjunction>()

/*eslint prefer-const: ["error", {"destructuring": "all"}]*/
let { from, to } = interval

const {
horizon = 6,
angularSeparationThreshold = ANGULAR_SEPARATION_THRESHOLD,
stepMinutes = 20
} = params

// Spica is close to the ecliptic, so we know that we may see it close to a planet:
// We are using the J2000.0 coordinates for Spica, we therefore need to convert them to the current epoch:
const spica = {
ra: 201.298,
dec: -11.1613
}

// Regulus is close to the ecliptic, so we know that we may see it close to a planet:
// We are using the J2000.0 coordinates for Regulus, we therefore need to convert them to the current epoch:
const regulus = {
ra: 152.093,
dec: 11.9672
}

while (from <= to) {
const moon = getLunarEquatorialCoordinate(from)

// Collate the positions of all planets other than Earth and those below the horizon in the sky:
// N.B. They may be in conjunction, but they won't be visible to our local observer if they are
// below the horizon.
const positions = [
...getPlanetaryPositions(from, observer),
{
name: 'Moon',
...moon,
...convertEquatorialToHorizontal(from, observer, moon)
},
{
name: 'Spica',
...spica,
...convertEquatorialToHorizontal(from, observer, spica)
},
{
name: 'Regulus',
...regulus,
...convertEquatorialToHorizontal(from, observer, regulus)
}
]

// Loop over all pairs of planets and check for conjunctions:
for (let i = 0; i < positions.length; i++) {
for (let j = i + 1; j < positions.length; j++) {
// If either of the planets is below the horizon, skip this pair:
if (positions[i].alt < horizon || positions[j].alt < horizon) continue

// Get the positions of the two planets:
const alterior = positions[i]
const ulterior = positions[j]

// Create a unique key for the conjunction between the two planets, sorted by name:
const key = [alterior.name, ulterior.name].sort().join('-')

// Check for a conjunction between the two planets by comparing their angular separation:
const separation = getAngularSeparation(
{
θ: positions[i].alt,
φ: positions[i].az
},
{
θ: positions[j].alt,
φ: positions[j].az
}
)

// Update the conjunction if the angular separation is less than the threshold,
// and the conjunction is the closest one found so far:
if (
isConjunction(from, [alterior, ulterior], {
horizon,
angularSeparationThreshold
}) &&
(!conjunctions.has(key) || conjunctions.get(key)!.angularSeparation > separation)
) {
const conjunction: Conjunction = {
datetime: from,
targets: [alterior, ulterior],
angularSeparation: separation,
ra: (alterior.ra + ulterior.ra) / 2,
dec: (alterior.dec + ulterior.dec) / 2
}

conjunctions.set(key, conjunction)
}
}
}

// Increment the from date by the step size:
from = new Date(from.getTime() + stepMinutes * 60000)
}

return conjunctions
}

/*****************************************************************************************************************/

0 comments on commit 0839756

Please sign in to comment.