- 新增日食、月食、本地可见性、中心线、半影区域、SVG 图示与沙罗周期信息 - 新增行星冲合、留、方照、物理星历、视直径、相位、亮肢角、轨道节点等计算 - 新增木星伽利略卫星位置、现象与接触事件计算 - 新增恒星星表、星座判定、自行修正与观测辅助能力 - 新增 coord、formula、orbit、sundial、lite/sun、lite/moon 等扩展包 - 完善农历年号、月相英文别名、视差角、大气质量、折射、日晷与双星计算 - 增加 NASA、JPL Horizons、IMCCE 等回归测试数据与基线测试 - 重构基础算法文件组织,补充大量公开 API 注释和语义回归测试 - 更新中文和英文 README,补充示例、精度说明、SVG 配图
202 lines
7.3 KiB
Go
202 lines
7.3 KiB
Go
package basic
|
||
|
||
import "math"
|
||
|
||
const solarRadiusAU = 695700.0 / astronomicalUnitKM
|
||
|
||
// JupiterGalileanPhenomenon 木星伽利略卫星瞬时现象 / instantaneous Galilean-satellite phenomena.
|
||
//
|
||
// Transit 表示卫星本体在木星盘前;Occultation 表示卫星在木星盘后被掩蔽;Eclipse 表示卫星落入木星本影;ShadowTransit 表示卫星影心落在可见木星盘面上。
|
||
// Transit means the satellite itself is in front of Jupiter's disk; Occultation means it is hidden behind the disk; Eclipse means the satellite lies in Jupiter's umbra; ShadowTransit means the center of the satellite shadow falls on the visible Jovian disk.
|
||
type JupiterGalileanPhenomenon struct {
|
||
Transit bool
|
||
Occultation bool
|
||
Eclipse bool
|
||
ShadowTransit bool
|
||
|
||
ShadowOffsetXArcsec float64
|
||
ShadowOffsetYArcsec float64
|
||
|
||
ShadowOffsetXJupiterRadii float64
|
||
ShadowOffsetYJupiterRadii float64
|
||
}
|
||
|
||
// JupiterGalileanSatellitePhenomenon 单颗伽利略卫星瞬时现象 / instantaneous phenomena of one Galilean satellite.
|
||
func JupiterGalileanSatellitePhenomenon(jd float64, satellite int) JupiterGalileanPhenomenon {
|
||
if satellite < 1 || satellite > 4 || !isFinite(jd) {
|
||
return invalidJupiterGalileanPhenomenon()
|
||
}
|
||
context := newJupiterGalileanObservationContext(jd)
|
||
return context.phenomenonForSatellite(satellite - 1)
|
||
}
|
||
|
||
// JupiterGalileanSatellitePhenomena 四颗伽利略卫星瞬时现象 / instantaneous phenomena of the four Galilean satellites.
|
||
func JupiterGalileanSatellitePhenomena(jd float64) [4]JupiterGalileanPhenomenon {
|
||
var phenomena [4]JupiterGalileanPhenomenon
|
||
context := newJupiterGalileanObservationContext(jd)
|
||
for i := range phenomena {
|
||
phenomena[i] = context.phenomenonForSatellite(i)
|
||
}
|
||
return phenomena
|
||
}
|
||
|
||
func (context jupiterGalileanObservationContext) phenomenonForSatellite(index int) JupiterGalileanPhenomenon {
|
||
if index < 0 || index >= 4 || context.jupiterDistance == 0 {
|
||
return invalidJupiterGalileanPhenomenon()
|
||
}
|
||
observation := context.observationForSatellite(index)
|
||
stateVector := Vector3{observation.State.X, observation.State.Y, observation.State.Z}
|
||
radiusAU := jupiterGalileanEquatorialRadiusKM / astronomicalUnitKM
|
||
xEarth := observation.OffsetXJupiterRadii
|
||
yEarth := observation.OffsetYJupiterRadii
|
||
onEarthDisk := ellipseInside(xEarth, yEarth, 1, context.earthMinorRadius)
|
||
|
||
xSunAU := vectorDot(stateVector, context.sunEast)
|
||
ySunAU := vectorDot(stateVector, context.sunNorth)
|
||
zSunAU := vectorDot(stateVector, context.sunLineOfSight)
|
||
xSun := xSunAU / radiusAU
|
||
ySun := ySunAU / radiusAU
|
||
umbraScale := jupiterUmbraScale(zSunAU, context.sunDistanceAU)
|
||
eclipse := false
|
||
if zSunAU > 0 && umbraScale > 0 {
|
||
eclipse = ellipseInside(xSun, ySun, umbraScale, context.sunMinorRadius*umbraScale)
|
||
}
|
||
|
||
shadowTransit, shadowXAU, shadowYAU := context.shadowTransitFor(stateVector)
|
||
phenomenon := JupiterGalileanPhenomenon{
|
||
Transit: onEarthDisk && observation.InFrontOfJupiter,
|
||
Occultation: onEarthDisk && !observation.InFrontOfJupiter,
|
||
Eclipse: eclipse,
|
||
ShadowTransit: shadowTransit,
|
||
|
||
ShadowOffsetXArcsec: math.NaN(),
|
||
ShadowOffsetYArcsec: math.NaN(),
|
||
ShadowOffsetXJupiterRadii: math.NaN(),
|
||
ShadowOffsetYJupiterRadii: math.NaN(),
|
||
}
|
||
if shadowTransit {
|
||
phenomenon.ShadowOffsetXArcsec = math.Atan2(shadowXAU, context.jupiterDistance) * deg * 3600
|
||
phenomenon.ShadowOffsetYArcsec = math.Atan2(shadowYAU, context.jupiterDistance) * deg * 3600
|
||
phenomenon.ShadowOffsetXJupiterRadii = shadowXAU / radiusAU
|
||
phenomenon.ShadowOffsetYJupiterRadii = shadowYAU / radiusAU
|
||
}
|
||
return phenomenon
|
||
}
|
||
|
||
func (context jupiterGalileanObservationContext) shadowTransitFor(stateVector Vector3) (bool, float64, float64) {
|
||
radiusAU := jupiterGalileanEquatorialRadiusKM / astronomicalUnitKM
|
||
satelliteBody := context.toBodyCoordinates(stateVector)
|
||
satelliteBody = Vector3{satelliteBody[0] / radiusAU, satelliteBody[1] / radiusAU, satelliteBody[2] / radiusAU}
|
||
directionBody := context.toBodyCoordinates(context.sunLineOfSight)
|
||
intersectionBody, ok := ellipsoidRayIntersection(satelliteBody, directionBody, jupiterPolarRadiusRatio())
|
||
if !ok {
|
||
return false, 0, 0
|
||
}
|
||
normalBody := Vector3{intersectionBody[0], intersectionBody[1], intersectionBody[2] / (jupiterPolarRadiusRatio() * jupiterPolarRadiusRatio())}
|
||
earthBody := context.toBodyCoordinates(context.earthDirection)
|
||
if vectorDot(normalBody, earthBody) <= 0 {
|
||
return false, 0, 0
|
||
}
|
||
intersection := context.fromBodyCoordinates(Vector3{
|
||
intersectionBody[0] * radiusAU,
|
||
intersectionBody[1] * radiusAU,
|
||
intersectionBody[2] * radiusAU,
|
||
})
|
||
xAU := vectorDot(intersection, context.east)
|
||
yAU := vectorDot(intersection, context.north)
|
||
x := xAU / radiusAU
|
||
y := yAU / radiusAU
|
||
if !ellipseInside(x, y, 1, context.earthMinorRadius) {
|
||
return false, 0, 0
|
||
}
|
||
return true, xAU, yAU
|
||
}
|
||
|
||
func (context jupiterGalileanObservationContext) toBodyCoordinates(vector Vector3) Vector3 {
|
||
return Vector3{
|
||
vectorDot(vector, context.bodyX),
|
||
vectorDot(vector, context.bodyY),
|
||
vectorDot(vector, context.bodyZ),
|
||
}
|
||
}
|
||
|
||
func (context jupiterGalileanObservationContext) fromBodyCoordinates(vector Vector3) Vector3 {
|
||
return Vector3{
|
||
context.bodyX[0]*vector[0] + context.bodyY[0]*vector[1] + context.bodyZ[0]*vector[2],
|
||
context.bodyX[1]*vector[0] + context.bodyY[1]*vector[1] + context.bodyZ[1]*vector[2],
|
||
context.bodyX[2]*vector[0] + context.bodyY[2]*vector[1] + context.bodyZ[2]*vector[2],
|
||
}
|
||
}
|
||
|
||
func jupiterProjectedMinorRadius(direction, pole Vector3) float64 {
|
||
sinBeta := vectorDot(direction, pole)
|
||
cos2Beta := 1 - sinBeta*sinBeta
|
||
if cos2Beta < 0 {
|
||
cos2Beta = 0
|
||
}
|
||
ratio := jupiterPolarRadiusRatio()
|
||
return math.Sqrt(sinBeta*sinBeta + ratio*ratio*cos2Beta)
|
||
}
|
||
|
||
func jupiterPolarRadiusRatio() float64 {
|
||
return jupiterPhysicalModel.polarRadius / jupiterPhysicalModel.equatorialRadius
|
||
}
|
||
|
||
func ellipseInside(x, y, major, minor float64) bool {
|
||
if major <= 0 || minor <= 0 {
|
||
return false
|
||
}
|
||
return (x*x)/(major*major)+(y*y)/(minor*minor) <= 1+1e-12
|
||
}
|
||
|
||
func ellipsoidRayIntersection(origin, direction Vector3, polarRatio float64) (Vector3, bool) {
|
||
invPolar2 := 1 / (polarRatio * polarRatio)
|
||
a := direction[0]*direction[0] + direction[1]*direction[1] + direction[2]*direction[2]*invPolar2
|
||
b := 2 * (origin[0]*direction[0] + origin[1]*direction[1] + origin[2]*direction[2]*invPolar2)
|
||
c := origin[0]*origin[0] + origin[1]*origin[1] + origin[2]*origin[2]*invPolar2 - 1
|
||
discriminant := b*b - 4*a*c
|
||
if discriminant < 0 {
|
||
return Vector3{}, false
|
||
}
|
||
sqrtDiscriminant := math.Sqrt(discriminant)
|
||
t1 := (-b - sqrtDiscriminant) / (2 * a)
|
||
t2 := (-b + sqrtDiscriminant) / (2 * a)
|
||
t := math.Inf(1)
|
||
if t1 > 0 {
|
||
t = t1
|
||
}
|
||
if t2 > 0 && t2 < t {
|
||
t = t2
|
||
}
|
||
if !isFinite(t) {
|
||
return Vector3{}, false
|
||
}
|
||
return Vector3{
|
||
origin[0] + t*direction[0],
|
||
origin[1] + t*direction[1],
|
||
origin[2] + t*direction[2],
|
||
}, true
|
||
}
|
||
|
||
func jupiterUmbraScale(distanceBehindAU, sunDistanceAU float64) float64 {
|
||
if distanceBehindAU <= 0 || sunDistanceAU <= 0 {
|
||
return 0
|
||
}
|
||
jupiterRadiusAU := jupiterGalileanEquatorialRadiusKM / astronomicalUnitKM
|
||
umbraLength := jupiterRadiusAU * sunDistanceAU / (solarRadiusAU - jupiterRadiusAU)
|
||
if umbraLength <= 0 {
|
||
return 0
|
||
}
|
||
return 1 - distanceBehindAU/umbraLength
|
||
}
|
||
|
||
func invalidJupiterGalileanPhenomenon() JupiterGalileanPhenomenon {
|
||
nan := math.NaN()
|
||
return JupiterGalileanPhenomenon{
|
||
ShadowOffsetXArcsec: nan,
|
||
ShadowOffsetYArcsec: nan,
|
||
ShadowOffsetXJupiterRadii: nan,
|
||
ShadowOffsetYJupiterRadii: nan,
|
||
}
|
||
}
|