2019-10-24 10:44:21 +08:00
|
|
|
|
package basic
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"math"
|
|
|
|
|
|
|
|
|
|
|
|
"b612.me/astro/planet"
|
|
|
|
|
|
. "b612.me/astro/tools"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneL(jd float64) float64 {
|
|
|
|
|
|
return planet.WherePlanet(7, 0, jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneB(jd float64) float64 {
|
|
|
|
|
|
return planet.WherePlanet(7, 1, jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
}
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneR(jd float64) float64 {
|
|
|
|
|
|
return planet.WherePlanet(7, 2, jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
}
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func ANeptuneX(jd float64) float64 {
|
|
|
|
|
|
l := NeptuneL(jd)
|
|
|
|
|
|
b := NeptuneB(jd)
|
|
|
|
|
|
r := NeptuneR(jd)
|
|
|
|
|
|
el := planet.WherePlanet(-1, 0, jd)
|
|
|
|
|
|
eb := planet.WherePlanet(-1, 1, jd)
|
|
|
|
|
|
er := planet.WherePlanet(-1, 2, jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
x := r*Cos(b)*Cos(l) - er*Cos(eb)*Cos(el)
|
|
|
|
|
|
return x
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func ANeptuneY(jd float64) float64 {
|
2019-10-24 10:44:21 +08:00
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
l := NeptuneL(jd)
|
|
|
|
|
|
b := NeptuneB(jd)
|
|
|
|
|
|
r := NeptuneR(jd)
|
|
|
|
|
|
el := planet.WherePlanet(-1, 0, jd)
|
|
|
|
|
|
eb := planet.WherePlanet(-1, 1, jd)
|
|
|
|
|
|
er := planet.WherePlanet(-1, 2, jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
y := r*Cos(b)*Sin(l) - er*Cos(eb)*Sin(el)
|
|
|
|
|
|
return y
|
|
|
|
|
|
}
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func ANeptuneZ(jd float64) float64 {
|
|
|
|
|
|
//l := NeptuneL(jd)
|
|
|
|
|
|
b := NeptuneB(jd)
|
|
|
|
|
|
r := NeptuneR(jd)
|
|
|
|
|
|
// el := planet.WherePlanet(-1, 0, jd)
|
|
|
|
|
|
eb := planet.WherePlanet(-1, 1, jd)
|
|
|
|
|
|
er := planet.WherePlanet(-1, 2, jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
z := r*Sin(b) - er*Sin(eb)
|
|
|
|
|
|
return z
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func ANeptuneXYZ(jd float64) (float64, float64, float64) {
|
|
|
|
|
|
l := NeptuneL(jd)
|
|
|
|
|
|
b := NeptuneB(jd)
|
|
|
|
|
|
r := NeptuneR(jd)
|
|
|
|
|
|
el := planet.WherePlanet(-1, 0, jd)
|
|
|
|
|
|
eb := planet.WherePlanet(-1, 1, jd)
|
|
|
|
|
|
er := planet.WherePlanet(-1, 2, jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
x := r*Cos(b)*Cos(l) - er*Cos(eb)*Cos(el)
|
|
|
|
|
|
y := r*Cos(b)*Sin(l) - er*Cos(eb)*Sin(el)
|
|
|
|
|
|
z := r*Sin(b) - er*Sin(eb)
|
|
|
|
|
|
return x, y, z
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneApparentRa(jd float64) float64 {
|
|
|
|
|
|
lo, bo := NeptuneApparentLoBo(jd)
|
|
|
|
|
|
eps := TrueObliquity(jd)
|
|
|
|
|
|
ra := math.Atan2((Sin(lo)*Cos(eps) - Tan(bo)*Sin(eps)), Cos(lo))
|
2019-10-24 10:44:21 +08:00
|
|
|
|
ra = ra * 180 / math.Pi
|
|
|
|
|
|
return Limit360(ra)
|
|
|
|
|
|
}
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneApparentDec(jd float64) float64 {
|
|
|
|
|
|
lo, bo := NeptuneApparentLoBo(jd)
|
|
|
|
|
|
eps := TrueObliquity(jd)
|
|
|
|
|
|
dec := ArcSin(Sin(bo)*Cos(eps) + Cos(bo)*Sin(eps)*Sin(lo))
|
2019-10-24 10:44:21 +08:00
|
|
|
|
return dec
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneApparentRaDec(jd float64) (float64, float64) {
|
|
|
|
|
|
lo, bo := NeptuneApparentLoBo(jd)
|
|
|
|
|
|
eps := TrueObliquity(jd)
|
|
|
|
|
|
ra := math.Atan2((Sin(lo)*Cos(eps) - Tan(bo)*Sin(eps)), Cos(lo))
|
2019-10-24 10:44:21 +08:00
|
|
|
|
ra = ra * 180 / math.Pi
|
2026-05-01 22:38:44 +08:00
|
|
|
|
dec := ArcSin(Sin(bo)*Cos(eps) + Cos(bo)*Sin(eps)*Sin(lo))
|
2019-10-24 10:44:21 +08:00
|
|
|
|
return Limit360(ra), dec
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func EarthNeptuneAway(jd float64) float64 {
|
|
|
|
|
|
x, y, z := ANeptuneXYZ(jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
to := math.Sqrt(x*x + y*y + z*z)
|
|
|
|
|
|
return to
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneApparentLo(jd float64) float64 {
|
|
|
|
|
|
x, y, z := ANeptuneXYZ(jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
to := 0.0057755183 * math.Sqrt(x*x+y*y+z*z)
|
2026-05-01 22:38:44 +08:00
|
|
|
|
x, y, z = ANeptuneXYZ(jd - to)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
lo := math.Atan2(y, x)
|
|
|
|
|
|
bo := math.Atan2(z, math.Sqrt(x*x+y*y))
|
|
|
|
|
|
lo = lo * 180 / math.Pi
|
|
|
|
|
|
bo = bo * 180 / math.Pi
|
|
|
|
|
|
lo = Limit360(lo)
|
2026-05-01 22:38:44 +08:00
|
|
|
|
//lo-=GXCLo(lo,bo,jd)/3600;
|
|
|
|
|
|
//bo+=GXCBo(lo,bo,jd);
|
|
|
|
|
|
lo += Nutation2000Bi(jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
return lo
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneApparentBo(jd float64) float64 {
|
|
|
|
|
|
x, y, z := ANeptuneXYZ(jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
to := 0.0057755183 * math.Sqrt(x*x+y*y+z*z)
|
2026-05-01 22:38:44 +08:00
|
|
|
|
x, y, z = ANeptuneXYZ(jd - to)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
//lo := math.Atan2(y, x)
|
|
|
|
|
|
bo := math.Atan2(z, math.Sqrt(x*x+y*y))
|
|
|
|
|
|
//lo = lo * 180 / math.Pi
|
|
|
|
|
|
bo = bo * 180 / math.Pi
|
2026-05-01 22:38:44 +08:00
|
|
|
|
//lo+=GXCLo(lo,bo,jd);
|
|
|
|
|
|
//bo+=GXCBo(lo,bo,jd)/3600;
|
|
|
|
|
|
//lo+=Nutation2000Bi(jd);
|
2019-10-24 10:44:21 +08:00
|
|
|
|
return bo
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneApparentLoBo(jd float64) (float64, float64) {
|
|
|
|
|
|
x, y, z := ANeptuneXYZ(jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
to := 0.0057755183 * math.Sqrt(x*x+y*y+z*z)
|
2026-05-01 22:38:44 +08:00
|
|
|
|
x, y, z = ANeptuneXYZ(jd - to)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
lo := math.Atan2(y, x)
|
|
|
|
|
|
bo := math.Atan2(z, math.Sqrt(x*x+y*y))
|
|
|
|
|
|
lo = lo * 180 / math.Pi
|
|
|
|
|
|
bo = bo * 180 / math.Pi
|
|
|
|
|
|
lo = Limit360(lo)
|
2026-05-01 22:38:44 +08:00
|
|
|
|
//lo-=GXCLo(lo,bo,jd)/3600;
|
|
|
|
|
|
//bo+=GXCBo(lo,bo,jd);
|
|
|
|
|
|
lo += Nutation2000Bi(jd)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
return lo, bo
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneMag(jd float64) float64 {
|
|
|
|
|
|
sunDistance := NeptuneR(jd)
|
|
|
|
|
|
earthDistance := EarthNeptuneAway(jd)
|
|
|
|
|
|
earthSunDistance := planet.WherePlanet(-1, 2, jd)
|
|
|
|
|
|
i := (sunDistance*sunDistance + earthDistance*earthDistance - earthSunDistance*earthSunDistance) / (2 * sunDistance * earthDistance)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
i = ArcCos(i)
|
2026-05-01 22:38:44 +08:00
|
|
|
|
mag := -6.87 + 5*math.Log10(sunDistance*earthDistance)
|
|
|
|
|
|
return FloatRound(mag, 2)
|
2019-10-24 10:44:21 +08:00
|
|
|
|
}
|
2022-05-16 20:42:15 +08:00
|
|
|
|
|
|
|
|
|
|
func NeptuneHeight(jde, lon, lat, timezone float64) float64 {
|
|
|
|
|
|
// 转换为世界时
|
|
|
|
|
|
utcJde := jde - timezone/24.0
|
|
|
|
|
|
// 计算视恒星时
|
|
|
|
|
|
ra, dec := NeptuneApparentRaDec(TD2UT(utcJde, true))
|
|
|
|
|
|
st := Limit360(ApparentSiderealTime(utcJde)*15 + lon)
|
|
|
|
|
|
// 计算时角
|
2026-05-01 22:38:44 +08:00
|
|
|
|
hourAngle := Limit360(st - ra)
|
2022-05-16 20:42:15 +08:00
|
|
|
|
// 高度角、时角与天球座标三角转换公式
|
2026-05-01 22:38:44 +08:00
|
|
|
|
// sin(h)=sin(lat)*sin(dec)+cos(dec)*cos(lat)*cos(hourAngle)
|
|
|
|
|
|
sinHeight := Sin(lat)*Sin(dec) + Cos(dec)*Cos(lat)*Cos(hourAngle)
|
2022-05-16 20:42:15 +08:00
|
|
|
|
return ArcSin(sinHeight)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func NeptuneAzimuth(jde, lon, lat, timezone float64) float64 {
|
|
|
|
|
|
// 转换为世界时
|
|
|
|
|
|
utcJde := jde - timezone/24.0
|
|
|
|
|
|
// 计算视恒星时
|
|
|
|
|
|
ra, dec := NeptuneApparentRaDec(TD2UT(utcJde, true))
|
|
|
|
|
|
st := Limit360(ApparentSiderealTime(utcJde)*15 + lon)
|
|
|
|
|
|
// 计算时角
|
2026-05-01 22:38:44 +08:00
|
|
|
|
hourAngle := Limit360(st - ra)
|
2022-05-16 20:42:15 +08:00
|
|
|
|
// 三角转换公式
|
2026-05-01 22:38:44 +08:00
|
|
|
|
tanAzimuth := Sin(hourAngle) / (Cos(hourAngle)*Sin(lat) - Tan(dec)*Cos(lat))
|
|
|
|
|
|
azimuth := ArcTan(tanAzimuth)
|
|
|
|
|
|
if azimuth < 0 {
|
|
|
|
|
|
if hourAngle/15 < 12 {
|
|
|
|
|
|
return azimuth + 360
|
2022-05-16 20:42:15 +08:00
|
|
|
|
}
|
2026-05-01 22:38:44 +08:00
|
|
|
|
return azimuth + 180
|
2022-05-16 20:42:15 +08:00
|
|
|
|
}
|
2026-05-01 22:38:44 +08:00
|
|
|
|
if hourAngle/15 < 12 {
|
|
|
|
|
|
return azimuth + 180
|
2022-05-16 20:42:15 +08:00
|
|
|
|
}
|
2026-05-01 22:38:44 +08:00
|
|
|
|
return azimuth
|
2022-05-16 20:42:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneHourAngle(jd, lon, timezone float64) float64 {
|
|
|
|
|
|
siderealLongitude := Limit360(ApparentSiderealTime(jd-timezone/24)*15 + lon)
|
|
|
|
|
|
hourAngle := siderealLongitude - NeptuneApparentRa(TD2UT(jd-timezone/24.0, true))
|
|
|
|
|
|
if hourAngle < 0 {
|
|
|
|
|
|
hourAngle += 360
|
2022-05-16 20:42:15 +08:00
|
|
|
|
}
|
2026-05-01 22:38:44 +08:00
|
|
|
|
return hourAngle
|
2022-05-16 20:42:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func NeptuneCulminationTime(jde, lon, timezone float64) float64 {
|
|
|
|
|
|
//jde 世界时,非力学时,当地时区 0时,无需转换力学时
|
|
|
|
|
|
//ra,dec 瞬时天球座标,非J2000等时间天球坐标
|
|
|
|
|
|
jde = math.Floor(jde) + 0.5
|
2026-05-01 22:38:44 +08:00
|
|
|
|
estimateJD := jde + Limit360(360-NeptuneHourAngle(jde, lon, timezone))/15.0/24.0*0.99726851851851851851
|
|
|
|
|
|
normalizedHourAngle := func(jde, lon, timezone float64) float64 {
|
|
|
|
|
|
currentHourAngle := NeptuneHourAngle(jde, lon, timezone)
|
|
|
|
|
|
if currentHourAngle < 180 {
|
|
|
|
|
|
currentHourAngle += 360
|
2022-05-16 20:42:15 +08:00
|
|
|
|
}
|
2026-05-01 22:38:44 +08:00
|
|
|
|
return currentHourAngle
|
2022-05-16 20:42:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
for {
|
2026-05-01 22:38:44 +08:00
|
|
|
|
prevJD := estimateJD
|
|
|
|
|
|
hourAngleDelta := normalizedHourAngle(prevJD, lon, timezone) - 360
|
|
|
|
|
|
hourAngleSlope := (normalizedHourAngle(prevJD+0.000005, lon, timezone) - normalizedHourAngle(prevJD-0.000005, lon, timezone)) / 0.00001
|
|
|
|
|
|
estimateJD = prevJD - hourAngleDelta/hourAngleSlope
|
|
|
|
|
|
if math.Abs(estimateJD-prevJD) <= 0.00001 {
|
2022-05-16 20:42:15 +08:00
|
|
|
|
break
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2026-05-01 22:38:44 +08:00
|
|
|
|
return estimateJD
|
2022-05-16 20:42:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneRiseTime(jd, lon, lat, timezone, aeroCorrection, observerHeight float64) (float64, error) {
|
|
|
|
|
|
return neptuneRiseDown(jd, lon, lat, timezone, aeroCorrection, observerHeight, true)
|
2022-05-16 20:42:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func NeptuneSetTime(jd, lon, lat, timezone, aeroCorrection, observerHeight float64) (float64, error) {
|
|
|
|
|
|
return neptuneRiseDown(jd, lon, lat, timezone, aeroCorrection, observerHeight, false)
|
2022-05-16 20:42:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 22:38:44 +08:00
|
|
|
|
func neptuneRiseDown(jd, lon, lat, timezone, aeroCorrection, observerHeight float64, isRise bool) (float64, error) {
|
|
|
|
|
|
return planetRiseDown(jd, lon, lat, timezone, aeroCorrection, observerHeight, isRise, NeptuneCulminationTime, NeptuneHeight, NeptuneApparentDec)
|
2022-05-16 20:42:15 +08:00
|
|
|
|
}
|