feat: 扩展天文计算能力
- 新增日食、月食、本地可见性、中心线、半影区域、SVG 图示与沙罗周期信息 - 新增行星冲合、留、方照、物理星历、视直径、相位、亮肢角、轨道节点等计算 - 新增木星伽利略卫星位置、现象与接触事件计算 - 新增恒星星表、星座判定、自行修正与观测辅助能力 - 新增 coord、formula、orbit、sundial、lite/sun、lite/moon 等扩展包 - 完善农历年号、月相英文别名、视差角、大气质量、折射、日晷与双星计算 - 增加 NASA、JPL Horizons、IMCCE 等回归测试数据与基线测试 - 重构基础算法文件组织,补充大量公开 API 注释和语义回归测试 - 更新中文和英文 README,补充示例、精度说明、SVG 配图
This commit is contained in:
@@ -0,0 +1,509 @@
|
||||
package basic
|
||||
|
||||
import (
|
||||
"math"
|
||||
|
||||
. "b612.me/astro/tools"
|
||||
)
|
||||
|
||||
// 太阳中天时刻,通过均时差计算
|
||||
func CulminationTime(jd, lon, tz float64) float64 { //实际中天时间
|
||||
jd = math.Floor(jd)
|
||||
tmp := (tz*15 - lon) * 4 / 60
|
||||
return jd + tmp/24.0 - SunTime(jd)/24.0
|
||||
}
|
||||
|
||||
func CulminationTimeN(jd, lon, tz float64, n int) float64 { //实际中天时间
|
||||
jd = math.Floor(jd)
|
||||
tmp := (tz*15 - lon) * 4 / 60
|
||||
return jd + tmp/24.0 - SunTimeN(jd, n)/24.0
|
||||
}
|
||||
|
||||
/*
|
||||
* 昏朦影传入 当天0时时刻
|
||||
*/
|
||||
func EveningTwilight(jd, lon, lat, tz, targetAltitude float64) (float64, error) {
|
||||
jd = math.Floor(jd) + 1.5
|
||||
localTimeZone := math.Round(lon / 15)
|
||||
culminationTime := CulminationTime(jd, lon, localTimeZone)
|
||||
if SunHeight(culminationTime, lon, lat, localTimeZone) < targetAltitude {
|
||||
return 0, ErrNeverRise
|
||||
}
|
||||
if SunHeight(culminationTime+0.5, lon, lat, localTimeZone) > targetAltitude {
|
||||
return 0, ErrNeverSet
|
||||
}
|
||||
tmp := (Sin(targetAltitude) - Sin(HSunApparentDec(culminationTime))*Sin(lat)) / (Cos(HSunApparentDec(culminationTime)) * Cos(lat))
|
||||
var sundown float64
|
||||
if math.Abs(tmp) <= 1 && lat < 85 {
|
||||
hourOffset := ArcCos(tmp) / 15
|
||||
sundown = culminationTime + hourOffset/24.0 + 35.0/24.0/60.0
|
||||
} else {
|
||||
sundown = culminationTime
|
||||
i := 0
|
||||
for LowSunHeight(sundown, lon, lat, localTimeZone) > targetAltitude {
|
||||
i++
|
||||
sundown += 15.0 / 60.0 / 24.0
|
||||
if i > 48 {
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
estimateJD := sundown - 5.00/24.00/60.00
|
||||
for {
|
||||
prevJD := estimateJD
|
||||
stDegree := SunHeight(prevJD, lon, lat, localTimeZone) - targetAltitude
|
||||
stDegreep := (SunHeight(prevJD+0.000005, lon, lat, localTimeZone) - SunHeight(prevJD-0.000005, lon, lat, localTimeZone)) / 0.00001
|
||||
estimateJD = prevJD - stDegree/stDegreep
|
||||
if math.Abs(estimateJD-prevJD) < 0.00001 {
|
||||
break
|
||||
}
|
||||
}
|
||||
return estimateJD - localTimeZone/24 + tz/24, nil
|
||||
}
|
||||
|
||||
func EveningTwilightN(jd, lon, lat, tz, targetAltitude float64, n int) (float64, error) {
|
||||
jd = math.Floor(jd) + 1.5
|
||||
localTimeZone := math.Round(lon / 15)
|
||||
culminationTime := CulminationTimeN(jd, lon, localTimeZone, n)
|
||||
if SunHeightN(culminationTime, lon, lat, localTimeZone, n) < targetAltitude {
|
||||
return 0, ErrNeverRise
|
||||
}
|
||||
if SunHeightN(culminationTime+0.5, lon, lat, localTimeZone, n) > targetAltitude {
|
||||
return 0, ErrNeverSet
|
||||
}
|
||||
tmp := (Sin(targetAltitude) - Sin(HSunApparentDecN(culminationTime, n))*Sin(lat)) / (Cos(HSunApparentDecN(culminationTime, n)) * Cos(lat))
|
||||
var sundown float64
|
||||
if math.Abs(tmp) <= 1 && lat < 85 {
|
||||
hourOffset := ArcCos(tmp) / 15
|
||||
sundown = culminationTime + hourOffset/24.0 + 35.0/24.0/60.0
|
||||
} else {
|
||||
sundown = culminationTime
|
||||
i := 0
|
||||
for lowSunHeightForN(sundown, lon, lat, localTimeZone, n) > targetAltitude {
|
||||
i++
|
||||
sundown += 15.0 / 60.0 / 24.0
|
||||
if i > 48 {
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
estimateJD := sundown - 5.00/24.00/60.00
|
||||
for {
|
||||
prevJD := estimateJD
|
||||
stDegree := SunHeightN(prevJD, lon, lat, localTimeZone, n) - targetAltitude
|
||||
stDegreep := (SunHeightN(prevJD+0.000005, lon, lat, localTimeZone, n) - SunHeightN(prevJD-0.000005, lon, lat, localTimeZone, n)) / 0.00001
|
||||
estimateJD = prevJD - stDegree/stDegreep
|
||||
if math.Abs(estimateJD-prevJD) < 0.00001 {
|
||||
break
|
||||
}
|
||||
}
|
||||
return estimateJD - localTimeZone/24 + tz/24, nil
|
||||
}
|
||||
|
||||
func MorningTwilight(jd, lon, lat, tz, targetAltitude float64) (float64, error) {
|
||||
// 调整到中午12点
|
||||
jd = math.Floor(jd) + 1.5
|
||||
|
||||
// 计算时区
|
||||
localTimeZone := math.Round(lon / 15)
|
||||
|
||||
// 计算太阳上中天时间
|
||||
culminationTime := CulminationTime(jd, lon, localTimeZone)
|
||||
|
||||
// 检查极夜和极昼条件
|
||||
if SunHeight(culminationTime, lon, lat, localTimeZone) < targetAltitude {
|
||||
return 0, ErrNeverRise
|
||||
}
|
||||
if SunHeight(culminationTime-0.5, lon, lat, localTimeZone) > targetAltitude {
|
||||
return 0, ErrNeverSet
|
||||
}
|
||||
|
||||
// 计算日出时间
|
||||
sunDec := HSunApparentDec(culminationTime)
|
||||
tmp := (Sin(targetAltitude) - Sin(sunDec)*Sin(lat)) / (Cos(sunDec) * Cos(lat))
|
||||
|
||||
var sunrise float64
|
||||
if math.Abs(tmp) <= 1 && lat < 85 {
|
||||
hourAngle := ArcCos(tmp) / 15
|
||||
sunrise = culminationTime - hourAngle/24 - 25.0/(24.0*60.0)
|
||||
} else {
|
||||
sunrise = culminationTime
|
||||
for i := 0; i < 48 && LowSunHeight(sunrise, lon, lat, localTimeZone) > targetAltitude; i++ {
|
||||
sunrise -= 15.0 / (60.0 * 24.0) // 每次减少15分钟
|
||||
}
|
||||
}
|
||||
|
||||
estimateJD := sunrise - 5.0/(24.0*60.0)
|
||||
for {
|
||||
prevJD := estimateJD
|
||||
heightDiff := SunHeight(prevJD, lon, lat, localTimeZone) - targetAltitude
|
||||
heightDerivative := (SunHeight(prevJD+0.000005, lon, lat, localTimeZone) - SunHeight(prevJD-0.000005, lon, lat, localTimeZone)) / 0.00001
|
||||
estimateJD = prevJD - heightDiff/heightDerivative
|
||||
|
||||
if math.Abs(estimateJD-prevJD) < 0.00001 {
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
return estimateJD - localTimeZone/24 + tz/24, nil
|
||||
}
|
||||
|
||||
func MorningTwilightN(jd, lon, lat, tz, targetAltitude float64, n int) (float64, error) {
|
||||
jd = math.Floor(jd) + 1.5
|
||||
localTimeZone := math.Round(lon / 15)
|
||||
culminationTime := CulminationTimeN(jd, lon, localTimeZone, n)
|
||||
if SunHeightN(culminationTime, lon, lat, localTimeZone, n) < targetAltitude {
|
||||
return 0, ErrNeverRise
|
||||
}
|
||||
if SunHeightN(culminationTime-0.5, lon, lat, localTimeZone, n) > targetAltitude {
|
||||
return 0, ErrNeverSet
|
||||
}
|
||||
|
||||
sunDec := HSunApparentDecN(culminationTime, n)
|
||||
tmp := (Sin(targetAltitude) - Sin(sunDec)*Sin(lat)) / (Cos(sunDec) * Cos(lat))
|
||||
|
||||
var sunrise float64
|
||||
if math.Abs(tmp) <= 1 && lat < 85 {
|
||||
hourAngle := ArcCos(tmp) / 15
|
||||
sunrise = culminationTime - hourAngle/24 - 25.0/(24.0*60.0)
|
||||
} else {
|
||||
sunrise = culminationTime
|
||||
for i := 0; i < 48 && lowSunHeightForN(sunrise, lon, lat, localTimeZone, n) > targetAltitude; i++ {
|
||||
sunrise -= 15.0 / (60.0 * 24.0)
|
||||
}
|
||||
}
|
||||
|
||||
estimateJD := sunrise - 5.0/(24.0*60.0)
|
||||
for {
|
||||
prevJD := estimateJD
|
||||
heightDiff := SunHeightN(prevJD, lon, lat, localTimeZone, n) - targetAltitude
|
||||
heightDerivative := (SunHeightN(prevJD+0.000005, lon, lat, localTimeZone, n) - SunHeightN(prevJD-0.000005, lon, lat, localTimeZone, n)) / 0.00001
|
||||
estimateJD = prevJD - heightDiff/heightDerivative
|
||||
|
||||
if math.Abs(estimateJD-prevJD) < 0.00001 {
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
return estimateJD - localTimeZone/24 + tz/24, nil
|
||||
}
|
||||
|
||||
/*
|
||||
* 太阳时角
|
||||
*/
|
||||
func SunTimeAngle(jd, lon, lat, tz float64) float64 {
|
||||
startime := Limit360(ApparentSiderealTime(jd-tz/24)*15 + lon)
|
||||
timeangle := startime - HSunApparentRa(TD2UT(jd-tz/24, true))
|
||||
if timeangle < 0 {
|
||||
timeangle += 360
|
||||
}
|
||||
return timeangle
|
||||
}
|
||||
|
||||
func SunTimeAngleN(jd, lon, lat, tz float64, n int) float64 {
|
||||
startime := Limit360(ApparentSiderealTime(jd-tz/24)*15 + lon)
|
||||
timeangle := startime - HSunApparentRaN(TD2UT(jd-tz/24, true), n)
|
||||
if timeangle < 0 {
|
||||
timeangle += 360
|
||||
}
|
||||
return timeangle
|
||||
}
|
||||
|
||||
// GetSunRiseTime 精确计算日出时间,传入当日0时JDE
|
||||
func GetSunRiseTime(julianDay, longitude, latitude, timeZone, zenithShift, height float64) (float64, error) {
|
||||
return calculateSunRiseSetTime(julianDay, longitude, latitude, timeZone, zenithShift, height, true)
|
||||
}
|
||||
|
||||
func GetSunRiseTimeN(julianDay, longitude, latitude, timeZone, zenithShift, height float64, n int) (float64, error) {
|
||||
return calculateSunRiseSetTimeN(julianDay, longitude, latitude, timeZone, zenithShift, height, true, n)
|
||||
}
|
||||
|
||||
// GetSunSetTime 精确计算日落时间,传入当日0时JDE
|
||||
func GetSunSetTime(julianDay, longitude, latitude, timeZone, zenithShift, height float64) (float64, error) {
|
||||
return calculateSunRiseSetTime(julianDay, longitude, latitude, timeZone, zenithShift, height, false)
|
||||
}
|
||||
|
||||
func GetSunSetTimeN(julianDay, longitude, latitude, timeZone, zenithShift, height float64, n int) (float64, error) {
|
||||
return calculateSunRiseSetTimeN(julianDay, longitude, latitude, timeZone, zenithShift, height, false, n)
|
||||
}
|
||||
|
||||
// calculateSunRiseSetTime 统一的日出日落计算函数
|
||||
func calculateSunRiseSetTime(julianDay, longitude, latitude, timeZone, zenithShift, height float64, isSunrise bool) (float64, error) {
|
||||
julianDay = math.Floor(julianDay) + 1.5
|
||||
naturalTimeZone := math.Round(longitude / 15)
|
||||
sunAngle := StandardAltitudeSun(zenithShift, height, latitude)
|
||||
|
||||
// 获取太阳上中天时间
|
||||
solarNoonTime := CulminationTime(julianDay, longitude, naturalTimeZone)
|
||||
|
||||
// 检查极夜极昼条件
|
||||
if err := checkPolarConditions(solarNoonTime, longitude, latitude, naturalTimeZone, sunAngle, isSunrise); err != nil {
|
||||
return 0, err
|
||||
}
|
||||
|
||||
// 计算初始估算时间
|
||||
initialTime := calculateInitialSunTime(solarNoonTime, longitude, latitude, naturalTimeZone, sunAngle, isSunrise)
|
||||
|
||||
// 牛顿-拉夫逊迭代求精确解
|
||||
return sunRiseSetNewtonRaphsonIteration(initialTime, longitude, latitude, naturalTimeZone, sunAngle, timeZone), nil
|
||||
}
|
||||
|
||||
func calculateSunRiseSetTimeN(julianDay, longitude, latitude, timeZone, zenithShift, height float64, isSunrise bool, n int) (float64, error) {
|
||||
julianDay = math.Floor(julianDay) + 1.5
|
||||
naturalTimeZone := math.Round(longitude / 15)
|
||||
sunAngle := StandardAltitudeSun(zenithShift, height, latitude)
|
||||
|
||||
solarNoonTime := CulminationTimeN(julianDay, longitude, naturalTimeZone, n)
|
||||
if err := checkPolarConditionsN(solarNoonTime, longitude, latitude, naturalTimeZone, sunAngle, isSunrise, n); err != nil {
|
||||
return 0, err
|
||||
}
|
||||
|
||||
initialTime := calculateInitialSunTimeN(solarNoonTime, longitude, latitude, naturalTimeZone, sunAngle, isSunrise, n)
|
||||
return sunRiseSetNewtonRaphsonIterationN(initialTime, longitude, latitude, naturalTimeZone, sunAngle, timeZone, n), nil
|
||||
}
|
||||
|
||||
// checkPolarConditions 检查极夜极昼条件
|
||||
func checkPolarConditions(solarNoonTime, longitude, latitude, naturalTimeZone, sunAngle float64, isSunrise bool) error {
|
||||
if SunHeight(solarNoonTime, longitude, latitude, naturalTimeZone) < sunAngle {
|
||||
return ErrNeverRise
|
||||
}
|
||||
|
||||
checkTime := solarNoonTime + 0.5
|
||||
if isSunrise {
|
||||
checkTime = solarNoonTime - 0.5
|
||||
}
|
||||
|
||||
if SunHeight(checkTime, longitude, latitude, naturalTimeZone) > sunAngle {
|
||||
return ErrNeverSet
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func checkPolarConditionsN(solarNoonTime, longitude, latitude, naturalTimeZone, sunAngle float64, isSunrise bool, n int) error {
|
||||
if SunHeightN(solarNoonTime, longitude, latitude, naturalTimeZone, n) < sunAngle {
|
||||
return ErrNeverRise
|
||||
}
|
||||
|
||||
checkTime := solarNoonTime + 0.5
|
||||
if isSunrise {
|
||||
checkTime = solarNoonTime - 0.5
|
||||
}
|
||||
|
||||
if SunHeightN(checkTime, longitude, latitude, naturalTimeZone, n) > sunAngle {
|
||||
return ErrNeverSet
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// calculateInitialSunTime 计算日出日落的初始估算时间
|
||||
func calculateInitialSunTime(solarNoonTime, longitude, latitude, naturalTimeZone, sunAngle float64, isSunrise bool) float64 {
|
||||
// 使用球面三角法计算: (sin(ho)-sin(φ)*sin(δ))/(cos(φ)*cos(δ))
|
||||
apparentDeclination := HSunApparentDec(solarNoonTime)
|
||||
cosHourAngle := (Sin(sunAngle) - Sin(apparentDeclination)*Sin(latitude)) / (Cos(apparentDeclination) * Cos(latitude))
|
||||
|
||||
if math.Abs(cosHourAngle) <= 1 && latitude < 85 {
|
||||
// 使用解析解
|
||||
hourAngle := ArcCos(cosHourAngle) / 15
|
||||
timeOffset := 25.0 / 24.0 / 60.0 // 日出偏移
|
||||
if !isSunrise {
|
||||
timeOffset = 35.0 / 24.0 / 60.0 // 日落偏移
|
||||
}
|
||||
|
||||
if isSunrise {
|
||||
return solarNoonTime - hourAngle/24 - timeOffset
|
||||
} else {
|
||||
return solarNoonTime + hourAngle/24 + timeOffset
|
||||
}
|
||||
} else {
|
||||
// 使用迭代逼近法(极地条件)
|
||||
return iterativeApproach(solarNoonTime, longitude, latitude, naturalTimeZone, sunAngle, isSunrise)
|
||||
}
|
||||
}
|
||||
|
||||
func calculateInitialSunTimeN(solarNoonTime, longitude, latitude, naturalTimeZone, sunAngle float64, isSunrise bool, n int) float64 {
|
||||
apparentDeclination := HSunApparentDecN(solarNoonTime, n)
|
||||
cosHourAngle := (Sin(sunAngle) - Sin(apparentDeclination)*Sin(latitude)) / (Cos(apparentDeclination) * Cos(latitude))
|
||||
|
||||
if math.Abs(cosHourAngle) <= 1 && latitude < 85 {
|
||||
hourAngle := ArcCos(cosHourAngle) / 15
|
||||
timeOffset := 25.0 / 24.0 / 60.0
|
||||
if !isSunrise {
|
||||
timeOffset = 35.0 / 24.0 / 60.0
|
||||
}
|
||||
|
||||
if isSunrise {
|
||||
return solarNoonTime - hourAngle/24 - timeOffset
|
||||
}
|
||||
return solarNoonTime + hourAngle/24 + timeOffset
|
||||
}
|
||||
|
||||
return iterativeApproachN(solarNoonTime, longitude, latitude, naturalTimeZone, sunAngle, isSunrise, n)
|
||||
}
|
||||
|
||||
// iterativeApproach 迭代逼近法计算(用于极地等特殊条件)
|
||||
func iterativeApproach(solarNoonTime, longitude, latitude, naturalTimeZone, sunAngle float64, isSunrise bool) float64 {
|
||||
estimatedTime := solarNoonTime
|
||||
stepSize := 15.0 / 60.0 / 24.0 // 15分钟步长
|
||||
if isSunrise {
|
||||
stepSize = -stepSize
|
||||
}
|
||||
|
||||
const maxIterations = 48
|
||||
for i := 0; i < maxIterations && LowSunHeight(estimatedTime, longitude, latitude, naturalTimeZone) > sunAngle; i++ {
|
||||
estimatedTime += stepSize
|
||||
}
|
||||
|
||||
return estimatedTime
|
||||
}
|
||||
|
||||
func iterativeApproachN(solarNoonTime, longitude, latitude, naturalTimeZone, sunAngle float64, isSunrise bool, n int) float64 {
|
||||
estimatedTime := solarNoonTime
|
||||
stepSize := 15.0 / 60.0 / 24.0
|
||||
if isSunrise {
|
||||
stepSize = -stepSize
|
||||
}
|
||||
|
||||
const maxIterations = 48
|
||||
for i := 0; i < maxIterations && lowSunHeightForN(estimatedTime, longitude, latitude, naturalTimeZone, n) > sunAngle; i++ {
|
||||
estimatedTime += stepSize
|
||||
}
|
||||
|
||||
return estimatedTime
|
||||
}
|
||||
|
||||
// sunRiseSetNewtonRaphsonIteration 牛顿-拉夫逊迭代法求精确解
|
||||
func sunRiseSetNewtonRaphsonIteration(initialTime, longitude, latitude, naturalTimeZone, sunAngle, timeZone float64) float64 {
|
||||
const (
|
||||
convergenceThreshold = 0.00001
|
||||
derivativeStep = 0.000005
|
||||
)
|
||||
|
||||
currentTime := initialTime
|
||||
|
||||
for {
|
||||
previousTime := currentTime
|
||||
|
||||
// 计算函数值:f(t) = SunHeight(t) - targetAngle
|
||||
functionValue := SunHeight(previousTime, longitude, latitude, naturalTimeZone) - sunAngle
|
||||
|
||||
// 计算导数:f'(t) ≈ (f(t+h) - f(t-h)) / (2h)
|
||||
derivative := (SunHeight(previousTime+derivativeStep, longitude, latitude, naturalTimeZone) -
|
||||
SunHeight(previousTime-derivativeStep, longitude, latitude, naturalTimeZone)) / (2 * derivativeStep)
|
||||
|
||||
// 牛顿-拉夫逊公式:t_new = t_old - f(t) / f'(t)
|
||||
currentTime = previousTime - functionValue/derivative
|
||||
|
||||
// 检查收敛
|
||||
if math.Abs(currentTime-previousTime) <= convergenceThreshold {
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
// 转换为指定时区
|
||||
return currentTime - naturalTimeZone/24 + timeZone/24
|
||||
}
|
||||
|
||||
func sunRiseSetNewtonRaphsonIterationN(initialTime, longitude, latitude, naturalTimeZone, sunAngle, timeZone float64, n int) float64 {
|
||||
const (
|
||||
convergenceThreshold = 0.00001
|
||||
derivativeStep = 0.000005
|
||||
)
|
||||
|
||||
currentTime := initialTime
|
||||
|
||||
for {
|
||||
previousTime := currentTime
|
||||
functionValue := SunHeightN(previousTime, longitude, latitude, naturalTimeZone, n) - sunAngle
|
||||
derivative := (SunHeightN(previousTime+derivativeStep, longitude, latitude, naturalTimeZone, n) -
|
||||
SunHeightN(previousTime-derivativeStep, longitude, latitude, naturalTimeZone, n)) / (2 * derivativeStep)
|
||||
currentTime = previousTime - functionValue/derivative
|
||||
if math.Abs(currentTime-previousTime) <= convergenceThreshold {
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
return currentTime - naturalTimeZone/24 + timeZone/24
|
||||
}
|
||||
|
||||
/*
|
||||
* 太阳高度角 世界时
|
||||
*/
|
||||
func SunHeight(jd, lon, lat, tz float64) float64 {
|
||||
//tmp := (tz*15 - lon) * 4 / 60
|
||||
//truejd := jd - tmp/24
|
||||
calcjd := jd - tz/24.0
|
||||
tjde := TD2UT(calcjd, true)
|
||||
st := Limit360(ApparentSiderealTime(calcjd)*15 + lon)
|
||||
ra, dec := HSunApparentRaDec(tjde)
|
||||
hourAngle := Limit360(st - ra)
|
||||
tmp2 := Sin(lat)*Sin(dec) + Cos(dec)*Cos(lat)*Cos(hourAngle)
|
||||
return ArcSin(tmp2)
|
||||
}
|
||||
|
||||
func SunHeightN(jd, lon, lat, tz float64, n int) float64 {
|
||||
calcjd := jd - tz/24.0
|
||||
tjde := TD2UT(calcjd, true)
|
||||
st := Limit360(ApparentSiderealTime(calcjd)*15 + lon)
|
||||
ra, dec := HSunApparentRaDecN(tjde, n)
|
||||
hourAngle := Limit360(st - ra)
|
||||
tmp2 := Sin(lat)*Sin(dec) + Cos(dec)*Cos(lat)*Cos(hourAngle)
|
||||
return ArcSin(tmp2)
|
||||
}
|
||||
|
||||
func LowSunHeight(jd, lon, lat, tz float64) float64 {
|
||||
//tmp := (tz*15 - lon) * 4 / 60
|
||||
//truejd := jd - tmp/24
|
||||
calcjd := jd - tz/24
|
||||
st := Limit360(ApparentSiderealTime(calcjd)*15 + lon)
|
||||
hourAngle := Limit360(st - SunApparentRa(TD2UT(calcjd, true)))
|
||||
dec := SunApparentDec(TD2UT(calcjd, true))
|
||||
tmp2 := Sin(lat)*Sin(dec) + Cos(dec)*Cos(lat)*Cos(hourAngle)
|
||||
return ArcSin(tmp2)
|
||||
}
|
||||
|
||||
func lowSunHeightForN(jd, lon, lat, tz float64, n int) float64 {
|
||||
if n < 0 {
|
||||
return LowSunHeight(jd, lon, lat, tz)
|
||||
}
|
||||
return SunHeightN(jd, lon, lat, tz, n)
|
||||
}
|
||||
|
||||
func SunAzimuth(jd, lon, lat, tz float64) float64 {
|
||||
//tmp := (tz*15 - lon) * 4 / 60
|
||||
//truejd := jd - tmp/24
|
||||
calcjd := jd - tz/24
|
||||
st := Limit360(ApparentSiderealTime(calcjd)*15 + lon)
|
||||
hourAngle := Limit360(st - HSunApparentRa(TD2UT(calcjd, true)))
|
||||
tmp2 := Sin(hourAngle) / (Cos(hourAngle)*Sin(lat) - Tan(HSunApparentDec(TD2UT(calcjd, true)))*Cos(lat))
|
||||
azimuth := ArcTan(tmp2)
|
||||
if azimuth < 0 {
|
||||
if hourAngle/15 < 12 {
|
||||
return azimuth + 360
|
||||
}
|
||||
return azimuth + 180
|
||||
}
|
||||
if hourAngle/15 < 12 {
|
||||
return azimuth + 180
|
||||
}
|
||||
return azimuth
|
||||
}
|
||||
|
||||
func SunAzimuthN(jd, lon, lat, tz float64, n int) float64 {
|
||||
calcjd := jd - tz/24
|
||||
st := Limit360(ApparentSiderealTime(calcjd)*15 + lon)
|
||||
hourAngle := Limit360(st - HSunApparentRaN(TD2UT(calcjd, true), n))
|
||||
tmp2 := Sin(hourAngle) / (Cos(hourAngle)*Sin(lat) - Tan(HSunApparentDecN(TD2UT(calcjd, true), n))*Cos(lat))
|
||||
azimuth := ArcTan(tmp2)
|
||||
if azimuth < 0 {
|
||||
if hourAngle/15 < 12 {
|
||||
return azimuth + 360
|
||||
}
|
||||
return azimuth + 180
|
||||
}
|
||||
if hourAngle/15 < 12 {
|
||||
return azimuth + 180
|
||||
}
|
||||
return azimuth
|
||||
}
|
||||
Reference in New Issue
Block a user