Title: | Sun Position and Daylight Calculations |
---|---|
Description: | Compute the position of the sun, and local solar time using Meeus' formulae. Compute day and/or night length using different twilight definitions or arbitrary sun elevation angles. This package is part of the 'r4photobiology' suite, Aphalo, P. J. (2015) <doi:10.19232/uv4pb.2015.1.14>. Algorithms from Meeus (1998, ISBN:0943396611). |
Authors: | Pedro J. Aphalo [aut, cre] |
Maintainer: | Pedro J. Aphalo <[email protected]> |
License: | GPL (>= 2) |
Version: | 0.1.2 |
Built: | 2025-01-09 16:30:33 UTC |
Source: | https://github.com/aphalo/SunCalcMeeus |
Compute the position of the sun, and local solar time using Meeus' formulae. Compute day and/or night length using different twilight definitions or arbitrary sun elevation angles. This package is part of the 'r4photobiology' suite, Aphalo, P. J. (2015) doi:10.19232/uv4pb.2015.1.14. Algorithms from Meeus (1998, ISBN:0943396611).
Please see the vignette 0: The R for Photobiology Suite for a description of the suite.
Maintainer: Pedro J. Aphalo [email protected] (ORCID)
Aphalo, Pedro J. (2015) The r4photobiology suite. UV4Plants Bulletin, 2015:1, 21-29. doi:10.19232/uv4pb.2015.1.14.
Useful links:
Report bugs at https://github.com/aphalo/SunCalcMeeus/issues
# daylength sunrise_time(lubridate::today(tzone = "EET"), tz = "EET", geocode = data.frame(lat = 60, lon = 25), unit.out = "hour") day_length(lubridate::today(tzone = "EET"), tz = "EET", geocode = data.frame(lat = 60, lon = 25), unit.out = "hour") sun_angles(lubridate::now(tzone = "EET"), tz = "EET", geocode = data.frame(lat = 60, lon = 25))
# daylength sunrise_time(lubridate::today(tzone = "EET"), tz = "EET", geocode = data.frame(lat = 60, lon = 25), unit.out = "hour") day_length(lubridate::today(tzone = "EET"), tz = "EET", geocode = data.frame(lat = 60, lon = 25), unit.out = "hour") sun_angles(lubridate::now(tzone = "EET"), tz = "EET", geocode = data.frame(lat = 60, lon = 25))
Convert a datetime into a time of day expressed in hours, minutes or seconds from midnight in local time for a time zone. This conversion is useful when time-series data for different days needs to be compared or plotted based on the local time-of-day.
as_tod(x, unit.out = "hours", tz = NULL)
as_tod(x, unit.out = "hours", tz = NULL)
x |
a datetime object accepted by lubridate functions. |
unit.out |
character string, One of "tod_time", "hours", "minutes", or "seconds". |
tz |
character string indicating time zone to be used in output. |
A numeric vector of the same length as x
. If
unit.out = "tod_time"
an object of class "tod_time"
which
a numeric vector as with unit.out = "hours"
but with the class
attribute set to "tod_time"
, which dispatches to special
format()
and print()
methods.
Other Time of day functions:
format.tod_time()
,
print.tod_time()
library(lubridate) my_instants <- ymd_hms("2020-05-17 12:05:03") + days(c(0, 30)) my_instants as_tod(my_instants) as_tod(my_instants, unit.out = "tod_time")
library(lubridate) my_instants <- ymd_hms("2020-05-17 12:05:03") + days(c(0, 30)) my_instants as_tod(my_instants) as_tod(my_instants, unit.out = "tod_time")
Convert a solar_time object into solar_date object
as.solar_date(x, time)
as.solar_date(x, time)
x |
solar_time object. |
time |
an R date time object that provides the date part. |
Objects of class "solar_time" lack date information, it describes
the time since local astronomical or true midnight. This function
adds the date information from the argument passed to time time
assembling a modified time
object of class "solar_date".
An object of class "solar.date" object derived from POSIXct. This is needed only for unambiguous formatting and printing.
Other Local solar time functions:
is.solar_time()
,
print.solar_time()
,
solar_time()
Functions for calculating the timing of solar positions, given geographical coordinates and dates. They can be also used to find the time for an arbitrary solar elevation between 90 and -90 degrees by supplying "twilight" angle(s) as argument.
day_night( date = lubridate::now(tzone = "UTC"), tz = ifelse(lubridate::is.Date(date), "UTC", lubridate::tz(date)), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "none", unit.out = "hours" ) day_night_fast(date, tz, geocode, twilight, unit.out) is_daytime( date = lubridate::now(tzone = "UTC"), tz = ifelse(lubridate::is.Date(date), "UTC", lubridate::tz(date)), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "none", unit.out = "hours" ) noon_time( date = lubridate::now(tzone = "UTC"), tz = lubridate::tz(date), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "none", unit.out = "datetime" ) sunrise_time( date = lubridate::now(tzone = "UTC"), tz = lubridate::tz(date), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "sunlight", unit.out = "datetime" ) sunset_time( date = lubridate::now(tzone = "UTC"), tz = lubridate::tz(date), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "sunlight", unit.out = "datetime" ) day_length( date = lubridate::now(tzone = "UTC"), tz = "UTC", geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "sunlight", unit.out = "hours" ) night_length( date = lubridate::now(tzone = "UTC"), tz = "UTC", geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "sunlight", unit.out = "hours" )
day_night( date = lubridate::now(tzone = "UTC"), tz = ifelse(lubridate::is.Date(date), "UTC", lubridate::tz(date)), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "none", unit.out = "hours" ) day_night_fast(date, tz, geocode, twilight, unit.out) is_daytime( date = lubridate::now(tzone = "UTC"), tz = ifelse(lubridate::is.Date(date), "UTC", lubridate::tz(date)), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "none", unit.out = "hours" ) noon_time( date = lubridate::now(tzone = "UTC"), tz = lubridate::tz(date), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "none", unit.out = "datetime" ) sunrise_time( date = lubridate::now(tzone = "UTC"), tz = lubridate::tz(date), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "sunlight", unit.out = "datetime" ) sunset_time( date = lubridate::now(tzone = "UTC"), tz = lubridate::tz(date), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "sunlight", unit.out = "datetime" ) day_length( date = lubridate::now(tzone = "UTC"), tz = "UTC", geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "sunlight", unit.out = "hours" ) night_length( date = lubridate::now(tzone = "UTC"), tz = "UTC", geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), twilight = "sunlight", unit.out = "hours" )
date |
"vector" of |
tz |
character vector indicating time zone to be used in output and to
interpret |
geocode |
data frame with one or more rows and variables lon and lat as numeric values (degrees). If present, address will be copied to the output. |
twilight |
character string, one of "none", "rim", "refraction",
"sunlight", "civil", "nautical", "astronomical", or a |
unit.out |
character string, One of "datetime", "day", "hour", "minute", or "second". |
Twilight names are interpreted as follows. "none": solar elevation =
0 degrees. "rim": upper rim of solar disk at the horizon or solar elevation
= -0.53 / 2. "refraction": solar elevation = 0 degrees + refraction
correction. "sunlight": upper rim of solar disk corrected for refraction,
which is close to the value used by the online NOAA Solar Calculator.
"civil": -6 degrees, "naval": -12 degrees, and "astronomical": -18 degrees.
Unit names for output are as follows: "day", "hours", "minutes" and
"seconds" times for sunrise and sunset are returned as times-of-day since
midnight expressed in the chosen unit. "date" or "datetime" return the same
times as datetime objects with TZ set (this is much slower than "hours").
Day length and night length are returned as numeric values expressed in
hours when ‘"datetime"’ is passed as argument to unit.out
. If
twilight is a numeric vector of length two, the element with index 1 is
used for sunrise and that with index 2 for sunset.
is_daytime()
supports twilight specifications by name, a test
like sun_elevation() > 0
may be used directly for a numeric angle.
A tibble with variables day, tz, twilight.rise, twilight.set, longitude, latitude, address, sunrise, noon, sunset, daylength, nightlength or the corresponding individual vectors.
The value returned represents an instant in time or a duration. The
class of the object returned varies depending on the argument passed to
parameter unit.out
. If unit.out = "datetime"
, the returned
value is a "POSIXct" vector, otherwise it is a "numeric" vector.
is_daytime()
returns a logical vector, with TRUE
for
day time and FALSE
for night time.
noon_time
, sunrise_time
and sunset_time
return a
vector of POSIXct times
day_length
and night_length
return numeric a vector
giving the length in hours
Be aware that R's Date
class does not save time zone
metadata. This can lead to ambiguities in the current implementation
based on time instants. The argument passed to date
should be
of class POSIXct
, in other words an instant in time, from which
the correct date will be computed based on the tz
argument.
The time zone in which times passed to date
as argument are
expressed does not need to be the local one or match the geocode, however,
the returned values will be in the same time zone as the input.
Function day_night()
is an implementation of Meeus equations as
used in NOAAs on-line web calculator, which are very precise and valid for
a very broad range of dates. For sunrise and sunset the times are affected
by refraction in the atmosphere, which does in turn depend on weather
conditions. The effect of refraction on the apparent position of the sun is
only an estimate based on "typical" conditions. The more tangential to the
horizon is the path of the sun, the larger the effect of refraction is on
the times of visual occlusion of the sun behind the horizon—i.e. the
largest timing errors occur at high latitudes. The computation is not
defined for latitudes 90 and -90 degrees, i.e. at the poles.
There exists a different R implementation of the same algorithms called
"AstroCalcPureR" available as function astrocalc4r
in package
'fishmethods'. Although the equations used are almost all the same, the
function signatures and which values are returned differ. In particular,
the implementation in 'photobiology' splits the calculation into two
separate functions, one returning angles at given instants in time, and a
separate one returning the timing of events for given dates. In
'fishmethods' (= 1.11-0) there is a bug in function astrocalc4r() that
affects sunrise and sunset times. The times returned by the functions in
package 'photobiology' have been validated against the NOAA base
implementation.
In the current implementation functions sunrise_time
,
noon_time
, sunset_time
, day_length
,
night_length
and is_daytime
are all wrappers
on day_night
, so if more than one quantity is needed it is
preferable to directly call day_night
and extract the different
components from the returned list.
night_length
returns the length of night-time conditions in one
day (00:00:00 to 23:59:59), rather than the length of the night between two
consecutive days.
The primary source for the algorithm used is the book: Meeus, J. (1998) Astronomical Algorithms, 2 ed., Willmann-Bell, Richmond, VA, USA. ISBN 978-0943396613.
A different implementation is available at https://github.com/NEFSC/READ-PDB-AstroCalc4R/ and in R paclage 'fishmethods'. In 'fishmethods' (= 1.11-0) there is a bug in function astrocalc4r() that affects sunrise and sunset times.
An interactive web page using the same algorithms is available at https://gml.noaa.gov/grad/solcalc/. There are small differences in the returned times compared to our function that seem to be related to the estimation of atmospheric refraction (about 0.1 degrees).
Other astronomy related functions:
format.solar_time()
,
sun_angles()
library(lubridate) my.geocode <- data.frame(lon = 24.93838, lat = 60.16986, address = "Helsinki, Finland") day_night(ymd("2015-05-30", tz = "EET"), geocode = my.geocode) day_night(ymd("2015-05-30", tz = "EET") + days(1:10), geocode = my.geocode, twilight = "civil") sunrise_time(ymd("2015-05-30", tz = "EET"), geocode = my.geocode) noon_time(ymd("2015-05-30", tz = "EET"), geocode = my.geocode) sunset_time(ymd("2015-05-30", tz = "EET"), geocode = my.geocode) day_length(ymd("2015-05-30", tz = "EET"), geocode = my.geocode) day_length(ymd("2015-05-30", tz = "EET"), geocode = my.geocode, unit.out = "day") is_daytime(ymd("2015-05-30", tz = "EET") + hours(c(0, 6, 12, 18, 24)), geocode = my.geocode) is_daytime(ymd_hms("2015-05-30 03:00:00", tz = "EET"), geocode = my.geocode) is_daytime(ymd_hms("2015-05-30 00:00:00", tz = "UTC"), geocode = my.geocode) is_daytime(ymd_hms("2015-05-30 03:00:00", tz = "EET"), geocode = my.geocode, twilight = "civil") is_daytime(ymd_hms("2015-05-30 00:00:00", tz = "UTC"), geocode = my.geocode, twilight = "civil")
library(lubridate) my.geocode <- data.frame(lon = 24.93838, lat = 60.16986, address = "Helsinki, Finland") day_night(ymd("2015-05-30", tz = "EET"), geocode = my.geocode) day_night(ymd("2015-05-30", tz = "EET") + days(1:10), geocode = my.geocode, twilight = "civil") sunrise_time(ymd("2015-05-30", tz = "EET"), geocode = my.geocode) noon_time(ymd("2015-05-30", tz = "EET"), geocode = my.geocode) sunset_time(ymd("2015-05-30", tz = "EET"), geocode = my.geocode) day_length(ymd("2015-05-30", tz = "EET"), geocode = my.geocode) day_length(ymd("2015-05-30", tz = "EET"), geocode = my.geocode, unit.out = "day") is_daytime(ymd("2015-05-30", tz = "EET") + hours(c(0, 6, 12, 18, 24)), geocode = my.geocode) is_daytime(ymd_hms("2015-05-30 03:00:00", tz = "EET"), geocode = my.geocode) is_daytime(ymd_hms("2015-05-30 00:00:00", tz = "UTC"), geocode = my.geocode) is_daytime(ymd_hms("2015-05-30 03:00:00", tz = "EET"), geocode = my.geocode, twilight = "civil") is_daytime(ymd_hms("2015-05-30 00:00:00", tz = "UTC"), geocode = my.geocode, twilight = "civil")
Format a solar_time
object for pretty printing
## S3 method for class 'solar_time' format(x, ..., sep = ":")
## S3 method for class 'solar_time' format(x, ..., sep = ":")
x |
an R object |
... |
ignored |
sep |
character used as separator |
A character string with the time formatted as "HH:MM:SS", where ":"
is the argument passed to sep
.
Other astronomy related functions:
day_night()
,
sun_angles()
Format a tod_time
object into a character string for pretty printing.
## S3 method for class 'tod_time' format(x, ..., sep = ":")
## S3 method for class 'tod_time' format(x, ..., sep = ":")
x |
an R object |
... |
ignored |
sep |
character used as separator |
A character string with the time formatted as "HH:MM:SS", where ":"
is the argument passed to sep
.
Other Time of day functions:
as_tod()
,
print.tod_time()
Estimate of down-welling solar (short wave) irradiance at the top of the
atmosphere above a location on Earth, computed based on angles, Sun-Earth
distance and the solar constant. Astronomical computations are done with
function sun_angles()
.
irrad_extraterrestrial( time = lubridate::now(tzone = "UTC"), tz = lubridate::tz(time), geocode = data.frame(lon = 0, lat = 51.5, address = "Greenwich"), solar.constant = "NASA" )
irrad_extraterrestrial( time = lubridate::now(tzone = "UTC"), tz = lubridate::tz(time), geocode = data.frame(lon = 0, lat = 51.5, address = "Greenwich"), solar.constant = "NASA" )
time |
A "vector" of POSIXct Time, with any valid time zone (TZ) is allowed, default is current time. |
tz |
character string indicating time zone to be used in output. |
geocode |
data frame with variables lon and lat as numeric values (degrees), nrow > 1, allowed. |
solar.constant |
numeric or character If character, "WMO" or "NASA", if numeric, an irradiance value in the same units as the value to be returned. |
Numeric vector of extraterrestrial irradiance (in W / m2 if solar constant is a character value).
Function sun_angles
.
library(lubridate) irrad_extraterrestrial(ymd_hm("2021-06-21 12:00", tz = "UTC")) irrad_extraterrestrial(ymd_hm("2021-12-21 20:00", tz = "UTC")) irrad_extraterrestrial(ymd_hm("2021-06-21 00:00", tz = "UTC") + hours(1:23))
library(lubridate) irrad_extraterrestrial(ymd_hm("2021-06-21 12:00", tz = "UTC")) irrad_extraterrestrial(ymd_hm("2021-12-21 20:00", tz = "UTC")) irrad_extraterrestrial(ymd_hm("2021-06-21 00:00", tz = "UTC") + hours(1:23))
Query class
is.solar_time(x) is.solar_date(x)
is.solar_time(x) is.solar_date(x)
x |
an R object. |
A logical value indicating if the object x
is of class
"solar_time"
or "solar_date"
, depending on the function.
Other Local solar time functions:
as.solar_date()
,
print.solar_time()
,
solar_time()
The object x
is printed and returned invisibly.
## S3 method for class 'solar_time' print(x, ...) ## S3 method for class 'solar_date' print(x, ...)
## S3 method for class 'solar_time' print(x, ...) ## S3 method for class 'solar_date' print(x, ...)
x |
an R object |
... |
passed to |
Returns object x
, invisibly.
Default is to print the underlying POSIXct or Date as a solar time.
Other Local solar time functions:
as.solar_date()
,
is.solar_time()
,
solar_time()
Defaults to print the underlying numeric
vector as a solar time.
## S3 method for class 'tod_time' print(x, ...)
## S3 method for class 'tod_time' print(x, ...)
x |
an R object |
... |
passed to |
Returns object x
, invisibly.
Other Time of day functions:
as_tod()
,
format.tod_time()
Approximate relative air mass (AM) computed from the sun's apparent or true position (sun elevation or sun zenith angle) or from geographic and time coordinates.
relative_AM( elevation.angle = NULL, zenith.angle = NULL, occluded.value = NA_real_ ) relative_AMt( elevation.angle = NULL, zenith.angle = NULL, occluded.value = NA_real_ ) relative_AM_geotime( time = lubridate::now(tzone = "UTC"), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), occluded.value = NA_real_ ) relative_AMt_geotime( time = lubridate::now(tzone = "UTC"), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), occluded.value = NA_real_ )
relative_AM( elevation.angle = NULL, zenith.angle = NULL, occluded.value = NA_real_ ) relative_AMt( elevation.angle = NULL, zenith.angle = NULL, occluded.value = NA_real_ ) relative_AM_geotime( time = lubridate::now(tzone = "UTC"), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), occluded.value = NA_real_ ) relative_AMt_geotime( time = lubridate::now(tzone = "UTC"), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), occluded.value = NA_real_ )
elevation.angle , zenith.angle
|
numeric vector Angle in degrees for the
sun position. An argument should be passed to one and only one of
|
occluded.value |
numeric Value to return when elevation angle is negative (sun below the horizon). |
time |
A "vector" of POSIXct Time, with any valid time zone (TZ) is allowed, default is current time. |
tz |
character string indicating time zone to be used in output. |
geocode |
data frame with variables lon and lat as numeric values (degrees), nrow > 1, allowed. |
Function relative_AM()
implements equation (3) in Kasten and
Young (1989). This equation is only an approximation to the tabulated
values in the same paper and based on the apparent position of the sun as
observed from Earth surface. relative_AMt()
implements equation (5)
in Young (1994). This equation is only an approximation to the tabulated
values based on the true or astronomical position of the sun.
In both cases returned values are rounded to three significant digits.
Function relative_AM_geotime()
is a wrapper on relative_AM()
that calls function sun_elevation()
to obtain the apparent position
of the sun from the geographic and time coordinates. Function
relative_AMt_geotime()
is a wrapper on relative_AMt()
that
calls function sun_elevation()
to obtain the true position of the
sun from the geographic and time coordinates. At very low sun elevations
the values returned by these two functions differ slightly because of the
use of different approximations to correct for atmospheric refraction.
A numeric vector with the relative air mass values.
Although relative air mass is not defined when the sun is not visible,
returning a value different from the default NA
might be useful in
some cases and made possible by passing an argument to parameter
occluded.value
.
F. Kasten, A. T. Young (1989) Revised optical air mass tables and approximation formula. Applied Optics, 28, 4735-4738. doi:10.1364/AO.28.004735.
Young, A. T. (1994) Air mass and refraction. Applied Optics, 33, 1108-1110. doi:10.1364/AO.33.001108
# using the apparent sun elevation relative_AM(elevation.angle = c(90, 60, 30, 1, -10)) relative_AM(elevation.angle = c(90, 60, 30, 1, -10), occluded.value = Inf) relative_AM(zenith.angle = 0) # using the true or astronomical sun elevation relative_AMt(elevation.angle = c(90, 60, 30, 1, -10)) relative_AMt(elevation.angle = c(90, 60, 30, 1, -10), occluded.value = Inf) relative_AMt(zenith.angle = 0) # Define example geographic and time coordinates baires.geo <- data.frame(lat = 34.60361, lon = -58.38139, address = "Buenos Aires") # using time and geographic coordinates library(lubridate) relative_AM_geotime(ymd_hms("2014-06-23 12:00:00", tz = "America/Argentina/Buenos_Aires"), geocode = baires.geo) relative_AMt_geotime(ymd_hms("2014-06-23 12:00:00", tz = "America/Argentina/Buenos_Aires"), geocode = baires.geo) relative_AM_geotime(ymd_hms("2014-06-23 12:00:00", tz = "America/Argentina/Buenos_Aires") + hours(0:12), geocode = baires.geo)
# using the apparent sun elevation relative_AM(elevation.angle = c(90, 60, 30, 1, -10)) relative_AM(elevation.angle = c(90, 60, 30, 1, -10), occluded.value = Inf) relative_AM(zenith.angle = 0) # using the true or astronomical sun elevation relative_AMt(elevation.angle = c(90, 60, 30, 1, -10)) relative_AMt(elevation.angle = c(90, 60, 30, 1, -10), occluded.value = Inf) relative_AMt(zenith.angle = 0) # Define example geographic and time coordinates baires.geo <- data.frame(lat = 34.60361, lon = -58.38139, address = "Buenos Aires") # using time and geographic coordinates library(lubridate) relative_AM_geotime(ymd_hms("2014-06-23 12:00:00", tz = "America/Argentina/Buenos_Aires"), geocode = baires.geo) relative_AMt_geotime(ymd_hms("2014-06-23 12:00:00", tz = "America/Argentina/Buenos_Aires"), geocode = baires.geo) relative_AM_geotime(ymd_hms("2014-06-23 12:00:00", tz = "America/Argentina/Buenos_Aires") + hours(0:12), geocode = baires.geo)
solar_time()
computes the time of day expressed in seconds since the
astronomical midnight using and instant in time and a geocode as input. Solar
time is useful when we want to plot data according to the local solar time
rather than the local time in use at a time zone. How the returned instant in
time is expressed depends on the argument passed to unit.out
.
solar_time( time = lubridate::now(), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), unit.out = "time" )
solar_time( time = lubridate::now(), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), unit.out = "time" )
time |
POSIXct Time, any valid time zone (TZ) is allowed, default is current time. |
geocode |
data frame with variables lon and lat as numeric values (degrees). |
unit.out |
character string, One of "datetime", "time", "hour", "minute", or "second". |
Solar time is determined by the position of the sun in the sky and it almost always differs from the time expressed in the local time coordinates in use. The differences can vary from a few minutes up to a couple of hours depending on the exact location within the time zone and the use or not of daylight saving time.
In all cases solar time is expressed as time since local astronomical
midnight and, thus, lacks date information. If unit.out = "time"
, a
numeric value in seconds with an additional class attribute
"solar_time"; if unit.out = "datetime"
, a "POSIXct" value in seconds
from midnight but with an additional class attribute "solar_date"; if
unit.out = "hour"
or unit.out = "minute"
or unit.out =
"second"
, a numeric value.
Returned values are computed based on the time zone of the argument for parameter time. In the case of solar time, this timezone does not affect the result. However, in the case of solar dates the date part may be off by one day, if the time zone does not match the coordinates of the geocode value provided as argument.
The algorithm is approximate, it calculates the difference between
local solar noon and noon in the time zone of time
and uses this
value for the whole day when converting times into solar time. Days are not
exactly 24 h long. Between successive days the shift is only a few seconds,
and this leads to a small jump at midnight.
Other Local solar time functions:
as.solar_date()
,
is.solar_time()
,
print.solar_time()
BA.geocode <- data.frame(lon = -58.38156, lat = -34.60368, address = "Buenos Aires, Argentina") sol_t <- solar_time(lubridate::dmy_hms("21/06/2016 10:00:00", tz = "UTC"), BA.geocode) sol_t class(sol_t) sol_d <- solar_time(lubridate::dmy_hms("21/06/2016 10:00:00", tz = "UTC"), BA.geocode, unit.out = "datetime") sol_d class(sol_d)
BA.geocode <- data.frame(lon = -58.38156, lat = -34.60368, address = "Buenos Aires, Argentina") sol_t <- solar_time(lubridate::dmy_hms("21/06/2016 10:00:00", tz = "UTC"), BA.geocode) sol_t class(sol_t) sol_d <- solar_time(lubridate::dmy_hms("21/06/2016 10:00:00", tz = "UTC"), BA.geocode, unit.out = "datetime") sol_d class(sol_d)
Function sun_angles()
returns the solar angles and Sun to Earth
relative distance for given times and locations using a very accurate
algorithm. Convenience functions sun_azimuth()
,
sun_elevation()
, sun_zenith_angle()
and
distance_to_sun()
are wrappers on sun_angles()
that return
individual vectors.
sun_angles( time = lubridate::now(tzone = "UTC"), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), use.refraction = FALSE ) sun_angles_fast(time, tz, geocode, use.refraction) sun_elevation( time = lubridate::now(), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), use.refraction = FALSE ) sun_zenith_angle( time = lubridate::now(), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), use.refraction = FALSE ) sun_azimuth( time = lubridate::now(), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), use.refraction = FALSE ) distance_to_sun( time = lubridate::now(), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), use.refraction = FALSE )
sun_angles( time = lubridate::now(tzone = "UTC"), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), use.refraction = FALSE ) sun_angles_fast(time, tz, geocode, use.refraction) sun_elevation( time = lubridate::now(), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), use.refraction = FALSE ) sun_zenith_angle( time = lubridate::now(), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), use.refraction = FALSE ) sun_azimuth( time = lubridate::now(), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), use.refraction = FALSE ) distance_to_sun( time = lubridate::now(), tz = lubridate::tz(time), geocode = tibble::tibble(lon = 0, lat = 51.5, address = "Greenwich"), use.refraction = FALSE )
time |
A "vector" of POSIXct Time, with any valid time zone (TZ) is allowed, default is current time. |
tz |
character string indicating time zone to be used in output. |
geocode |
data frame with variables lon and lat as numeric values (degrees), nrow > 1, allowed. |
use.refraction |
logical Flag indicating whether to correct for fraction in the atmosphere. |
This function is an implementation of Meeus equations as used in NOAA's on-line web calculator, which are precise and valid for a very broad range of dates (years -1000 to 3000 at least). The apparent solar elevations near sunrise and sunset are affected by refraction in the atmosphere, which does in turn depend on weather conditions. The effect of refraction on the apparent position of the sun is only an estimate based on "typical" conditions for the atmosphere. The computation is not defined for latitudes 90 and -90 degrees, i.e. exactly at the poles. The function is vectorized and in particular passing a vector of times for a single geocode enhances performance very much as the equation of time, the most time consuming step, is computed only once.
For improved performance, if more than one angle is needed it
is preferable to directly call sun_angles
instead of the wrapper
functions as this avoids the unnecesary recalculation.
A data.frame
with variables time
(in same TZ as input),
TZ
, solartime
, longitude
, latitude
,
address
, azimuth
, elevation
, declination
,
eq.of.time
, hour.angle
, and distance
. If a data frame
with multiple rows is passed to geocode
and a vector of times longer
than one is passed to time
, sun position for all combinations of
locations and times are returned by sun_angles
. Angles are expressed
in degrees, solartime
is a vector of class "solar.time"
,
distance
is expressed in relative sun units.
Given an instant in time and a time zone, the date is
computed from these, and may differ by one day to that at the location
pointed by geocode
at the same instant in time, unless the argument
passed to tz
matches the time zone at this location.
There exists a different R implementation of the same algorithms called
"AstroCalcPureR" available as function astrocalc4r
in package
'fishmethods'. Although the equations used are almost all the same, the
function signatures and which values are returned differ. In particular,
the present implementation splits the calculation into two separate
functions, one returning angles at given instants in time, and a separate
one returning the timing of events for given dates.
The primary source for the algorithm used is the book: Meeus, J. (1998) Astronomical Algorithms, 2 ed., Willmann-Bell, Richmond, VA, USA. ISBN 978-0943396613.
A different implementation is available at https://github.com/NEFSC/READ-PDB-AstroCalc4R/.
An interactive web page using the same algorithms is available at https://gml.noaa.gov/grad/solcalc/. There are small differences in the returned times compared to our function that seem to be related to the estimation of atmospheric refraction (about 0.1 degrees).
Other astronomy related functions:
day_night()
,
format.solar_time()
library(lubridate) sun_angles() sun_azimuth() sun_elevation() sun_zenith_angle() sun_angles(ymd_hms("2014-09-23 12:00:00")) sun_angles(ymd_hms("2014-09-23 12:00:00"), geocode = data.frame(lat=60, lon=0)) sun_angles(ymd_hms("2014-09-23 12:00:00") + minutes((0:6) * 10))
library(lubridate) sun_angles() sun_azimuth() sun_elevation() sun_zenith_angle() sun_angles(ymd_hms("2014-09-23 12:00:00")) sun_angles(ymd_hms("2014-09-23 12:00:00"), geocode = data.frame(lat=60, lon=0)) sun_angles(ymd_hms("2014-09-23 12:00:00") + minutes((0:6) * 10))
Returns the difference in local time expressed in hours between two time zones at a given instant in time. The difference due to daylight saving time or Summer and Winter time as well as historical changes in time zones are taken into account.
tz_time_diff( when = lubridate::now(), tz.target = lubridate::tz(when), tz.reference = "UTC" )
tz_time_diff( when = lubridate::now(), tz.target = lubridate::tz(when), tz.reference = "UTC" )
when |
datetime A time instant |
tz.target , tz.reference
|
character Two time zones using names recognized by functions from package 'lubridate' |
A numeric
value.
This function is implemented using functions from package 'lubridate'.
For details on the handling of time zones, please, consult the
documentation for Sys.timezone
about system differences in
time zone names and handling.
Test validity of a geocode or ensure that a geocode is valid.
validate_geocode(geocode) is_valid_geocode(geocode) length_geocode(geocode) na_geocode()
validate_geocode(geocode) is_valid_geocode(geocode) length_geocode(geocode) na_geocode()
geocode |
data.frame with geocode data in columns |
validate_geocode
Converts to tibble, checks data bounds, converts
address to character if it is not already a character vector, or add
character NAs if the address column is missing.
is_valid_geocode
Checks if a geocode is valid, returning 0L if not,
and the number of row otherwise.
A valid geocode stored in a tibble.
FALSE for invalid, TRUE for valid.
FALSE for invalid, number of rows for valid.
A geo_code tibble with all fields set to suitable NAs.
validate_geocode(NA) validate_geocode(data.frame(lon = -25, lat = 66)) is_valid_geocode(NA) is_valid_geocode(1L) is_valid_geocode(data.frame(lon = -25, lat = 66)) na_geocode()
validate_geocode(NA) validate_geocode(data.frame(lon = -25, lat = 66)) is_valid_geocode(NA) is_valid_geocode(1L) is_valid_geocode(data.frame(lon = -25, lat = 66)) na_geocode()