2013-09-01 20:30:51 +00:00
# include "swe-glib.h"
# include "gswe-types.h"
2013-09-01 11:52:18 +00:00
# include "gswe-moment.h"
2013-09-02 21:12:56 +00:00
# include "swe-glib-private.h"
2013-09-01 11:52:18 +00:00
2013-09-05 10:25:58 +00:00
# include "../swe/src/swephexp.h"
2013-09-01 20:30:51 +00:00
2013-09-02 21:12:56 +00:00
# define SYNODIC 29.53058867
2013-09-01 11:52:18 +00:00
# define GSWE_MOMENT_GET_PRIVATE(obj) (G_TYPE_INSTANCE_GET_PRIVATE((obj), GSWE_TYPE_MOMENT, GsweMomentPrivate))
2013-09-01 20:30:51 +00:00
/**
* GsweMomentPrivate :
* @ timestamp : a # GsweTimestmp object representing the current local time at
* the given position specified by @ coordinates . Be warned though ,
* that the time zone is NOT checked against the coordinates !
* @ coordinates : the coordinates of the observers position
2013-09-11 01:32:49 +00:00
* @ house_system : the house system this object uses
2013-09-01 20:30:51 +00:00
* @ revision : an internal counter which is incremented whenever the timestamp
2013-09-11 01:32:49 +00:00
* or the coordinates change . When this number changes , every data
* that has a * _revision value here will be recalculated before the
* data is served
* @ house_list : ( element - type GsweHouseData ) : the list of house data
* @ house_revision : the revision of the calculated house data
* @ planet_list : ( element - type GswePlanetData ) : the list of planets
* @ points_revision : the revision of the points
* @ element_points : the table of the element points
* @ quality_points : the table of the quality points
* @ moon_phase_revision : the revision of the Moon phase data
* @ moon_phase : the calculated Moon phase data
* @ aspect_list : ( element - type GsweAspectData ) : the list of calculated aspects
* @ aspect_revision : the revision of the aspect data
* @ mirrorpoint_list : ( element - type GsweMirrorData ) : the list of calculated
* mirrorpoints ( antiscia )
* @ mirrorpoint_revision : the revision of the mirrorpoint data
2013-09-01 20:30:51 +00:00
*/
2013-09-01 11:52:18 +00:00
struct _GsweMomentPrivate {
2013-09-01 12:31:23 +00:00
GsweTimestamp * timestamp ;
2013-09-01 11:52:18 +00:00
GsweCoordinates coordinates ;
2013-09-01 20:30:51 +00:00
GsweHouseSystem house_system ;
guint revision ;
GList * house_list ;
guint house_revision ;
2013-09-02 14:19:45 +00:00
GList * planet_list ;
2013-09-02 21:10:35 +00:00
guint points_revision ;
GHashTable * element_points ;
GHashTable * quality_points ;
2013-09-02 21:12:56 +00:00
guint moon_phase_revision ;
GsweMoonPhaseData moon_phase ;
2013-09-02 23:11:18 +00:00
GList * aspect_list ;
guint aspect_revision ;
2013-09-03 10:59:32 +00:00
GList * mirrorpoint_list ;
guint mirrorpoint_revision ;
2013-09-01 11:52:18 +00:00
} ;
enum {
2013-09-01 12:18:57 +00:00
SIGNAL_CHANGED ,
2013-09-01 11:52:18 +00:00
SIGNAL_LAST
} ;
enum {
PROP_0 ,
PROP_TIMESTAMP
} ;
2013-09-02 23:11:18 +00:00
struct GsweAspectFinder {
GswePlanet planet1 ;
GswePlanet planet2 ;
} ;
2013-09-01 11:52:18 +00:00
static guint gswe_moment_signals [ SIGNAL_LAST ] = { 0 } ;
static void gswe_moment_dispose ( GObject * gobject ) ;
static void gswe_moment_finalize ( GObject * gobject ) ;
static void gswe_moment_set_property ( GObject * object , guint prop_id , const GValue * value , GParamSpec * pspec ) ;
static void gswe_moment_get_property ( GObject * object , guint prop_id , GValue * value , GParamSpec * pspec ) ;
2013-09-11 01:34:07 +00:00
static GsweCoordinates * gswe_coordinates_copy ( GsweCoordinates * coordinates ) ;
2013-09-01 11:52:18 +00:00
G_DEFINE_TYPE ( GsweMoment , gswe_moment , G_TYPE_OBJECT ) ;
static void
gswe_moment_class_init ( GsweMomentClass * klass )
{
GObjectClass * gobject_class = G_OBJECT_CLASS ( klass ) ;
g_type_class_add_private ( klass , sizeof ( GsweMomentPrivate ) ) ;
gobject_class - > dispose = gswe_moment_dispose ;
gobject_class - > finalize = gswe_moment_finalize ;
gobject_class - > set_property = gswe_moment_set_property ;
gobject_class - > get_property = gswe_moment_get_property ;
/**
2013-09-01 12:18:57 +00:00
* GsweMoment : : changed :
2013-09-01 11:52:18 +00:00
* @ moment : the GsweMoment object that received the signal
*
2013-09-01 12:18:57 +00:00
* The : : changed signal is emitted each time the time or coordinates are changed
2013-09-01 11:52:18 +00:00
*/
2013-09-10 23:54:36 +00:00
gswe_moment_signals [ SIGNAL_CHANGED ] = g_signal_new ( " changed " , G_OBJECT_CLASS_TYPE ( gobject_class ) , G_SIGNAL_RUN_FIRST , 0 , NULL , NULL , g_cclosure_marshal_generic , G_TYPE_NONE , 0 ) ;
2013-09-01 11:52:18 +00:00
/**
* GsweMoment : timestamp :
*
* The timestamp associated with this moment
*/
2013-09-01 12:31:23 +00:00
g_object_class_install_property ( gobject_class , PROP_TIMESTAMP , g_param_spec_object ( " timestamp " , " Timestamp " , " Timestamp of this moment " , GSWE_TYPE_TIMESTAMP , G_PARAM_READWRITE ) ) ;
2013-09-01 11:52:18 +00:00
}
static void
2013-09-01 12:18:57 +00:00
gswe_moment_emit_changed ( GsweMoment * moment )
2013-09-01 11:52:18 +00:00
{
2013-09-01 12:18:57 +00:00
g_signal_emit ( moment , gswe_moment_signals [ SIGNAL_CHANGED ] , 0 ) ;
2013-09-01 11:52:18 +00:00
}
void
2013-09-01 12:31:23 +00:00
gswe_moment_init ( GsweMoment * moment )
2013-09-01 11:52:18 +00:00
{
2013-09-01 12:31:23 +00:00
moment - > priv = GSWE_MOMENT_GET_PRIVATE ( moment ) ;
2013-09-01 11:52:18 +00:00
2013-09-01 12:31:23 +00:00
moment - > priv - > timestamp = NULL ;
2013-09-01 20:30:51 +00:00
moment - > priv - > house_list = NULL ;
moment - > priv - > planet_list = NULL ;
2013-09-02 23:11:18 +00:00
moment - > priv - > aspect_list = NULL ;
2013-09-03 10:59:32 +00:00
moment - > priv - > mirrorpoint_list = NULL ;
2013-09-02 21:10:35 +00:00
moment - > priv - > element_points = g_hash_table_new_full ( g_direct_hash , g_direct_equal , NULL , NULL ) ;
moment - > priv - > quality_points = g_hash_table_new_full ( g_direct_hash , g_direct_equal , NULL , NULL ) ;
moment - > priv - > house_revision = 0 ;
moment - > priv - > points_revision = 0 ;
2013-09-02 21:12:56 +00:00
moment - > priv - > moon_phase_revision = 0 ;
2013-09-02 23:11:18 +00:00
moment - > priv - > aspect_revision = 0 ;
2013-09-03 10:59:32 +00:00
moment - > priv - > mirrorpoint_revision = 0 ;
2013-09-01 20:30:51 +00:00
moment - > priv - > revision = 1 ;
2013-09-01 12:31:23 +00:00
}
static void
2013-09-09 08:51:50 +00:00
gswe_moment_timestamp_changed ( GsweTimestamp * timestamp , GsweMoment * moment )
2013-09-01 12:31:23 +00:00
{
2013-09-01 20:30:51 +00:00
moment - > priv - > revision + + ;
gswe_moment_emit_changed ( moment ) ;
2013-09-01 11:52:18 +00:00
}
static void
gswe_moment_dispose ( GObject * gobject )
{
2013-09-01 12:31:23 +00:00
GsweMoment * moment = GSWE_MOMENT ( gobject ) ;
g_signal_handlers_disconnect_by_func ( moment - > priv - > timestamp , gswe_moment_timestamp_changed , NULL ) ;
2013-09-01 11:52:18 +00:00
2013-09-01 12:31:23 +00:00
g_clear_object ( & moment - > priv - > timestamp ) ;
2013-09-01 11:52:18 +00:00
G_OBJECT_CLASS ( gswe_moment_parent_class ) - > dispose ( gobject ) ;
}
static void
gswe_moment_finalize ( GObject * gobject )
{
2013-09-01 20:30:51 +00:00
GsweMoment * moment = GSWE_MOMENT ( gobject ) ;
2013-09-01 11:52:18 +00:00
2013-09-01 20:30:51 +00:00
g_list_free_full ( moment - > priv - > house_list , g_free ) ;
2013-09-02 21:10:35 +00:00
g_list_free_full ( moment - > priv - > planet_list , g_free ) ;
2013-09-01 11:52:18 +00:00
G_OBJECT_CLASS ( gswe_moment_parent_class ) - > finalize ( gobject ) ;
}
static void
gswe_moment_set_property ( GObject * object , guint prop_id , const GValue * value , GParamSpec * pspec )
{
GsweMoment * moment = GSWE_MOMENT ( object ) ;
switch ( prop_id ) {
case PROP_TIMESTAMP :
gswe_moment_set_timestamp ( moment , ( g_value_get_object ( value ) ) ) ;
break ;
default :
G_OBJECT_WARN_INVALID_PROPERTY_ID ( object , prop_id , pspec ) ;
break ;
}
}
static void
gswe_moment_get_property ( GObject * object , guint prop_id , GValue * value , GParamSpec * pspec )
{
GsweMoment * moment = GSWE_MOMENT ( object ) ;
GsweMomentPrivate * priv = moment - > priv ;
switch ( prop_id ) {
case PROP_TIMESTAMP :
g_value_set_object ( value , priv - > timestamp ) ;
break ;
default :
G_OBJECT_WARN_INVALID_PROPERTY_ID ( object , prop_id , pspec ) ;
break ;
}
}
2013-09-02 14:17:23 +00:00
/**
* gswe_moment_set_timestamp :
* @ moment : A GsweMoment object
* @ timestamp : A GsweTimestamp object . The moment object holds a reference on
* timestamp , which is cleared when a new timestamp is set , or the
* moment object is disposed .
*
* Sets a new timestamp for this planetary moment . Also emits the : : changed
* signal to notify owner of this change . This helps redrawing screen data
* according to the new time value .
*/
2013-09-01 11:52:18 +00:00
void
gswe_moment_set_timestamp ( GsweMoment * moment , GsweTimestamp * timestamp )
{
2013-09-01 12:31:23 +00:00
if ( moment - > priv - > timestamp ! = NULL ) {
g_signal_handlers_disconnect_by_func ( moment - > priv - > timestamp , gswe_moment_timestamp_changed , NULL ) ;
g_clear_object ( & moment - > priv - > timestamp ) ;
}
2013-09-11 01:34:43 +00:00
moment - > priv - > revision + + ;
2013-09-01 12:31:23 +00:00
moment - > priv - > timestamp = timestamp ;
g_object_ref ( timestamp ) ;
2013-09-09 08:51:50 +00:00
g_signal_connect ( G_OBJECT ( timestamp ) , " changed " , G_CALLBACK ( gswe_moment_timestamp_changed ) , moment ) ;
2013-09-01 11:52:18 +00:00
2013-09-01 12:18:57 +00:00
/* Emit the changed signal to notify registrants of the change */
gswe_moment_emit_changed ( moment ) ;
2013-09-01 11:52:18 +00:00
}
2013-09-01 12:31:23 +00:00
/**
* gswe_moment_get_timestamp :
2013-09-11 01:28:15 +00:00
* @ moment : a GsweMoment object
2013-09-01 12:31:23 +00:00
*
2013-09-11 01:28:15 +00:00
* Gets the # GsweTimestamp object associated with @ moment
2013-09-02 14:17:23 +00:00
*
2013-09-11 01:28:15 +00:00
* Returns : ( transfer none ) : the GsweTimestamp object associated with @ moment .
* The returned object is used by @ moment exclusively . If you need it even
* after @ moment is destroyed , call g_object_ref ( ) on it .
2013-09-01 12:31:23 +00:00
*/
GsweTimestamp *
gswe_moment_get_timestamp ( GsweMoment * moment )
{
return moment - > priv - > timestamp ;
}
2013-09-11 01:34:07 +00:00
/**
* gswe_moment_set_coordinates :
* @ moment : a GsweMoment
* @ longitude : the longitude part of the coordinates , in degrees
* @ latitude : the latitude part of the coordinates , in degrees
* @ altitude : the altitude part of the coordinates , in meters . As also noted in
* the README , it is safe to pass a value of around 400.0 , unless
* you want to create a * really * precise chart
*
* Sets the coordinates associated with @ moment . Emits the : : changed signal on
* @ moment . All values depending on the coordinates ( planetary and house cusp
* positions , aspects , mirrorpoints , so basically everything ) should be
* re - fetched after changing it .
*/
void
gswe_moment_set_coordinates ( GsweMoment * moment , gdouble longitude , gdouble latitude , gdouble altitude )
{
moment - > priv - > coordinates . longitude = longitude ;
moment - > priv - > coordinates . latitude = latitude ;
moment - > priv - > coordinates . altitude = altitude ;
moment - > priv - > revision + + ;
gswe_moment_emit_changed ( moment ) ;
}
/**
* gswe_moment_get_coordinates :
* @ moment : a GsweMoment
*
* Gets the coordinates associated with @ moment .
*
* Returns : ( transfer full ) : a newly allocated GsweCoordinates structure with
* the coordinates associated with @ moment . The returned pointer
* should be freed with g_free if you don ' t need it any more .
*/
GsweCoordinates *
gswe_moment_get_coordinates ( GsweMoment * moment )
{
return gswe_coordinates_copy ( & ( moment - > priv - > coordinates ) ) ;
}
/**
* gswe_moment_error_quark :
*
* Returns the # GQuark that will be used for # GError values returned by the
* SWE - GLib API .
*
* Returns : a GQuark used to identify errors coming from the SWE - GLib API
*/
2013-09-01 11:52:18 +00:00
GQuark
gswe_moment_error_quark ( void )
{
return g_quark_from_static_string ( " swe-glib-gswe-moment-error " ) ;
}
2013-09-11 01:35:45 +00:00
/**
* gswe_moment_new :
*
* Creates a new , empty GsweMoment object . The object created this way can not
* be used for any calculations yet , you need to call various gswe_moment_set_ *
* functions first . It is preferred to call gswe_moment_new_full ( ) instead .
*
* Returns : a new GsweMoment object
*/
2013-09-01 11:52:18 +00:00
GsweMoment *
gswe_moment_new ( void )
{
return ( GsweMoment * ) g_object_new ( GSWE_TYPE_MOMENT , NULL ) ;
}
2013-09-11 01:35:45 +00:00
/**
* gswe_moment_new_full :
* @ timestamp : a # GsweTimestamp , the exact time of your calculations
* @ longitude : the longitude part of the observer ' s position , in degrees
* @ latitude : the latitude part of the observer ' s position , in degrees
* @ altitude : the altitude part of the coordinates , in meters . As also noted in
* the README , it is safe to pass a value of around 400.0 , unless
* you want to create a * really * precise chart
* @ house_system : the house system you want to use . WARNING ! Using GSWE_HOUSE_SYSTEM_NONE is currently a bad idea , the results are unpredicted
*
* Creates a new GsweMoment object with the timestamp , coordinates and house system set . This is the preferred way to create a GsweMoment object .
*
* Returns : a new GsweMoment object , which is usable out of the box
*/
2013-09-01 20:30:51 +00:00
GsweMoment *
gswe_moment_new_full ( GsweTimestamp * timestamp , gdouble longitude , gdouble latitude , gdouble altitude , GsweHouseSystem house_system )
{
GsweMoment * moment = gswe_moment_new ( ) ;
moment - > priv - > timestamp = timestamp ;
g_object_ref ( timestamp ) ;
2013-09-09 08:51:50 +00:00
g_signal_connect ( G_OBJECT ( timestamp ) , " changed " , G_CALLBACK ( gswe_moment_timestamp_changed ) , moment ) ;
2013-09-01 20:30:51 +00:00
moment - > priv - > coordinates . longitude = longitude ;
moment - > priv - > coordinates . latitude = latitude ;
moment - > priv - > coordinates . altitude = altitude ;
moment - > priv - > house_system = house_system ;
2013-09-11 01:36:19 +00:00
if ( house_system = = GSWE_HOUSE_SYSTEM_NONE ) {
g_warning ( " Using GSWE_HOUSE_SYSTEM_NONE is unsafe. You have been warned! " ) ;
}
2013-09-01 20:30:51 +00:00
return moment ;
}
2013-09-02 14:19:45 +00:00
static gint
find_by_planet_id ( gconstpointer a , gconstpointer b )
{
const GswePlanetData * planet_data = a ;
const GswePlanet * planet = b ;
if ( planet_data - > planet_id = = * planet ) {
return 0 ;
}
return 1 ;
}
static void
2013-09-10 23:08:35 +00:00
gswe_calculate_data_by_position ( GsweMoment * moment , GswePlanet planet , gdouble position , GError * * err )
2013-09-02 14:19:45 +00:00
{
GswePlanetData * planet_data = ( GswePlanetData * ) ( g_list_find_custom ( moment - > priv - > planet_list , & planet , find_by_planet_id ) - > data ) ;
GsweZodiac sign ;
GsweSignInfo * sign_info ;
if ( planet_data = = NULL ) {
return ;
}
if ( planet_data - > revision = = moment - > priv - > revision ) {
return ;
}
sign = ( GsweZodiac ) ceil ( position / 30.0 ) ;
if ( ( sign_info = g_hash_table_lookup ( gswe_sign_info_table , GINT_TO_POINTER ( sign ) ) ) = = NULL ) {
g_error ( " Calculations brought an unknown sign! " ) ;
}
planet_data - > position = position ;
planet_data - > retrograde = FALSE ;
2013-09-10 23:08:35 +00:00
planet_data - > house = gswe_moment_get_house ( moment , position , err ) ;
2013-09-02 14:19:45 +00:00
planet_data - > sign = sign_info ;
planet_data - > revision = moment - > priv - > revision ;
}
2013-09-01 20:30:51 +00:00
static void
2013-09-10 23:08:35 +00:00
gswe_moment_calculate_house_positions ( GsweMoment * moment , GError * * err )
2013-09-01 20:30:51 +00:00
{
gdouble cusps [ 13 ] ,
2013-09-10 23:08:35 +00:00
ascmc [ 10 ] ,
jd ;
2013-09-01 20:30:51 +00:00
gint i ;
GsweHouseSystemInfo * house_system_data ;
2013-09-02 14:19:45 +00:00
if ( moment - > priv - > house_revision = = moment - > priv - > revision ) {
return ;
}
2013-09-01 20:30:51 +00:00
if ( ( house_system_data = g_hash_table_lookup ( gswe_house_system_info_table , GINT_TO_POINTER ( moment - > priv - > house_system ) ) ) = = NULL ) {
2013-09-10 23:08:35 +00:00
g_set_error ( err , GSWE_MOMENT_ERROR , GSWE_MOMENT_ERROR_UNKNOWN_HSYS , " Unknown house system " ) ;
return ;
}
jd = gswe_timestamp_get_julian_day ( moment - > priv - > timestamp , err ) ;
if ( ( err ) & & ( * err ) ) {
return ;
2013-09-01 20:30:51 +00:00
}
2013-09-10 23:08:35 +00:00
swe_houses ( jd , moment - > priv - > coordinates . latitude , moment - > priv - > coordinates . longitude , house_system_data - > sweph_id , cusps , ascmc ) ;
2013-09-01 20:30:51 +00:00
g_list_free_full ( moment - > priv - > house_list , g_free ) ;
moment - > priv - > house_list = NULL ;
2013-09-10 23:08:35 +00:00
/* TODO: SWE house system 'G' (Gauquelin sector cusps) have 36 houses; we
* should detect that somehow ( house system ' G ' is not implemented yet in
* GsweHouseSystem , and all other house systems have exactly 12 houses , so
* this should not cause trouble yet , though ) */
2013-09-01 20:30:51 +00:00
for ( i = 12 ; i > = 1 ; i - - ) {
2013-09-02 21:29:58 +00:00
GsweHouseData * house_data = g_new0 ( GsweHouseData , 1 ) ;
2013-09-01 20:30:51 +00:00
2013-09-02 21:29:58 +00:00
house_data - > house = i ;
house_data - > cusp_position = cusps [ i ] ;
if ( ( house_data - > sign = g_hash_table_lookup ( gswe_sign_info_table , GINT_TO_POINTER ( ( gint ) ceilf ( cusps [ i ] / 30.0 ) ) ) ) = = NULL ) {
2013-09-10 23:08:35 +00:00
g_list_free_full ( moment - > priv - > house_list , g_free ) ;
g_set_error ( err , GSWE_MOMENT_ERROR , GSWE_MOMENT_ERROR_UNKNOWN_SIGN , " Calculation brought an unknown sign " ) ;
return ;
2013-09-02 21:29:58 +00:00
}
moment - > priv - > house_list = g_list_prepend ( moment - > priv - > house_list , house_data ) ;
2013-09-01 20:30:51 +00:00
}
moment - > priv - > house_revision = moment - > priv - > revision ;
2013-09-02 14:19:45 +00:00
if ( gswe_moment_has_planet ( moment , GSWE_PLANET_ASCENDENT ) ) {
2013-09-10 23:08:35 +00:00
gswe_calculate_data_by_position ( moment , GSWE_PLANET_ASCENDENT , ascmc [ 0 ] , err ) ;
2013-09-02 14:19:45 +00:00
}
if ( gswe_moment_has_planet ( moment , GSWE_PLANET_MC ) ) {
2013-09-10 23:08:35 +00:00
gswe_calculate_data_by_position ( moment , GSWE_PLANET_MC , ascmc [ 1 ] , err ) ;
2013-09-02 14:19:45 +00:00
}
if ( gswe_moment_has_planet ( moment , GSWE_PLANET_VERTEX ) ) {
2013-09-10 23:08:35 +00:00
gswe_calculate_data_by_position ( moment , GSWE_PLANET_VERTEX , ascmc [ 3 ] , err ) ;
2013-09-02 14:19:45 +00:00
}
2013-09-01 20:30:51 +00:00
}
2013-09-05 17:50:57 +00:00
/**
* gswe_moment_get_house_cusps :
* @ moment : The GsweMoment object to operate on
2013-09-10 23:55:10 +00:00
* @ err : a # GError
2013-09-05 17:50:57 +00:00
*
* Calculate house cusp positions based on the house system , location and time set in @ moment .
*
* Returns : ( element - type GsweHouseData ) ( transfer none ) : a GList of # GsweHouseData
*/
2013-09-01 20:30:51 +00:00
GList *
2013-09-10 23:08:35 +00:00
gswe_moment_get_house_cusps ( GsweMoment * moment , GError * * err )
2013-09-01 20:30:51 +00:00
{
if ( moment - > priv - > house_revision ! = moment - > priv - > revision ) {
2013-09-10 23:08:35 +00:00
gswe_moment_calculate_house_positions ( moment , err ) ;
2013-09-01 20:30:51 +00:00
}
return moment - > priv - > house_list ;
}
2013-09-11 01:35:45 +00:00
/**
* gswe_moment_has_planet :
* @ moment : a GsweMoment
* @ planet : the planet whose existence is queried
*
* Checks if @ planet is added to @ moment , e . g . its position and related data is calculated .
*
* Returns : # TRUE if @ planet is already added to @ moment , # FALSE otherwise
*/
2013-09-02 14:19:45 +00:00
gboolean
gswe_moment_has_planet ( GsweMoment * moment , GswePlanet planet )
{
return ( g_list_find_custom ( moment - > priv - > planet_list , & planet , find_by_planet_id ) ! = NULL ) ;
}
2013-09-11 01:35:45 +00:00
/**
* gswe_moment_add_planet :
* @ moment : a GsweMoment object
* @ planet : the planet to add
*
* Adds @ planet to the calculated planets of @ moment .
*/
2013-09-02 14:19:45 +00:00
void
gswe_moment_add_planet ( GsweMoment * moment , GswePlanet planet )
{
GswePlanetData * planet_data = g_new0 ( GswePlanetData , 1 ) ;
GswePlanetInfo * planet_info ;
if ( gswe_moment_has_planet ( moment , planet ) ) {
return ;
}
if ( ( planet_info = g_hash_table_lookup ( gswe_planet_info_table , GINT_TO_POINTER ( planet ) ) ) = = NULL ) {
g_warning ( " Unknown planet ID: %d " , planet ) ;
return ;
}
planet_data - > planet_id = planet ;
planet_data - > planet_info = planet_info ;
planet_data - > position = 0.0 ;
planet_data - > house = 1 ;
planet_data - > sign = NULL ;
planet_data - > revision = 0 ;
moment - > priv - > planet_list = g_list_append ( moment - > priv - > planet_list , planet_data ) ;
}
static void
planet_add ( gpointer key , gpointer value , gpointer user_data )
{
GswePlanet planet = ( GswePlanet ) GPOINTER_TO_INT ( key ) ;
GsweMoment * moment = GSWE_MOMENT ( user_data ) ;
gswe_moment_add_planet ( moment , planet ) ;
}
2013-09-11 01:35:45 +00:00
/**
* gswe_moment_add_all_planets :
* @ moment : a GsweMoment object
*
* Adds all known planets to @ moment .
*/
2013-09-02 14:19:45 +00:00
void
gswe_moment_add_all_planets ( GsweMoment * moment )
{
g_hash_table_foreach ( gswe_planet_info_table , planet_add , moment ) ;
}
2013-09-02 21:10:35 +00:00
static void
2013-09-10 23:08:35 +00:00
gswe_moment_calculate_planet ( GsweMoment * moment , GswePlanet planet , GError * * err )
2013-09-02 21:10:35 +00:00
{
GswePlanetData * planet_data = ( GswePlanetData * ) ( g_list_find_custom ( moment - > priv - > planet_list , & planet , find_by_planet_id ) - > data ) ;
gchar serr [ AS_MAXCH ] ;
gint ret ;
2013-09-10 23:08:35 +00:00
gdouble x2 [ 6 ] ,
jd ;
GError * calc_err = NULL ;
2013-09-02 21:10:35 +00:00
if ( planet_data = = NULL ) {
return ;
}
if ( planet_data - > revision = = moment - > priv - > revision ) {
return ;
}
if ( planet_data - > planet_info - > real_body = = FALSE ) {
g_warning ( " The position data of planet %d can not be calculated by this function " , planet ) ;
return ;
}
swe_set_topo ( moment - > priv - > coordinates . longitude , moment - > priv - > coordinates . latitude , moment - > priv - > coordinates . altitude ) ;
2013-09-10 23:08:35 +00:00
jd = gswe_timestamp_get_julian_day ( moment - > priv - > timestamp , err ) ;
if ( ( err ) & & ( * err ) ) {
return ;
}
if ( ( ret = swe_calc ( jd , planet_data - > planet_info - > sweph_id , SEFLG_SPEED | SEFLG_TOPOCTR , x2 , serr ) ) < 0 ) {
g_set_error ( err , GSWE_MOMENT_ERROR , GSWE_MOMENT_ERROR_SWE_ERROR_FATAL , " Swiss Ephemeris error: %s " , serr ) ;
2013-09-02 21:10:35 +00:00
return ;
} else if ( ret ! = ( SEFLG_SPEED | SEFLG_TOPOCTR ) ) {
2013-09-10 23:08:35 +00:00
g_set_error ( err , GSWE_MOMENT_ERROR , GSWE_MOMENT_ERROR_SWE_ERROR_NONFATAL , " Swiss Ephemeris error: %s " , serr ) ;
2013-09-02 21:10:35 +00:00
}
2013-09-10 23:08:35 +00:00
gswe_calculate_data_by_position ( moment , planet , x2 [ 0 ] , & calc_err ) ;
if ( calc_err ! = NULL ) {
g_clear_error ( err ) ;
if ( err ) {
* err = calc_err ;
}
}
2013-09-02 21:10:35 +00:00
planet_data - > retrograde = ( x2 [ 3 ] < 0 ) ;
}
2013-09-10 23:08:35 +00:00
static void
calculate_planet ( GswePlanetData * planet_data , GsweMoment * moment )
2013-09-02 21:10:35 +00:00
{
2013-09-10 23:08:35 +00:00
gswe_moment_calculate_planet ( moment , planet_data - > planet_id , NULL ) ;
2013-09-02 21:10:35 +00:00
}
2013-09-10 23:08:35 +00:00
static void
2013-09-02 21:10:35 +00:00
gswe_moment_calculate_all_planets ( GsweMoment * moment )
{
2013-09-10 23:08:35 +00:00
g_list_foreach ( moment - > priv - > planet_list , ( GFunc ) calculate_planet , moment ) ;
2013-09-02 21:10:35 +00:00
}
2013-09-05 17:50:57 +00:00
/**
* gswe_moment_get_planets :
* @ moment : The GsweMoment to operate on
*
* Get all the planets added to @ moment .
*
* Returns : ( element - type GswePlanetData ) ( transfer none ) : A # GList of # GswePlanetData .
*/
2013-09-02 14:19:45 +00:00
GList *
gswe_moment_get_planets ( GsweMoment * moment )
{
return moment - > priv - > planet_list ;
}
gint
2013-09-10 23:08:35 +00:00
gswe_moment_get_house ( GsweMoment * moment , gdouble position , GError * * err )
2013-09-02 14:19:45 +00:00
{
gint i ;
2013-09-11 01:36:19 +00:00
if ( moment - > priv - > house_system = = GSWE_HOUSE_SYSTEM_NONE ) {
return 0 ;
}
2013-09-10 23:08:35 +00:00
gswe_moment_calculate_house_positions ( moment , err ) ;
2013-09-02 14:19:45 +00:00
2013-09-10 23:08:35 +00:00
/* TODO: SWE house system 'G' (Gauquelin sector cusps) have 36 houses; we
* should detect that somehow ( house system ' G ' is not implemented yet in
* GsweHouseSystem , and all other house systems have exactly 12 houses , so
* this should not cause trouble yet , though ) */
2013-09-02 14:19:45 +00:00
for ( i = 1 ; i < = 12 ; i + + ) {
gint j = ( i < 12 ) ? i + 1 : 1 ;
gdouble cusp_i = * ( gdouble * ) g_list_nth_data ( moment - > priv - > house_list , i - 1 ) ,
cusp_j = * ( gdouble * ) g_list_nth_data ( moment - > priv - > house_list , j - 1 ) ;
if ( cusp_j < cusp_i ) {
if ( ( position > = cusp_i ) | | ( position < cusp_j ) ) {
return i ;
}
} else {
if ( ( position > = cusp_i ) & & ( position < cusp_j ) ) {
return i ;
}
}
}
return 0 ;
}
2013-09-02 21:10:35 +00:00
GswePlanetData *
2013-09-10 23:08:35 +00:00
gswe_moment_get_planet ( GsweMoment * moment , GswePlanet planet , GError * * err )
2013-09-02 14:19:45 +00:00
{
GswePlanetData * planet_data = ( GswePlanetData * ) ( g_list_find_custom ( moment - > priv - > planet_list , & planet , find_by_planet_id ) - > data ) ;
if ( planet_data = = NULL ) {
2013-09-10 23:08:35 +00:00
g_set_error ( err , GSWE_MOMENT_ERROR , GSWE_MOMENT_ERROR_NONADDED_PLANET , " Specified planet is not added to the moment object " ) ;
2013-09-02 21:10:35 +00:00
return NULL ;
2013-09-02 14:19:45 +00:00
}
2013-09-10 23:08:35 +00:00
gswe_moment_calculate_planet ( moment , planet , err ) ;
2013-09-02 14:19:45 +00:00
2013-09-02 21:10:35 +00:00
return planet_data ;
}
2013-09-02 14:19:45 +00:00
2013-09-02 21:10:35 +00:00
static void
add_points ( GswePlanetData * planet_data , GsweMoment * moment )
{
guint point ;
2013-09-02 14:19:45 +00:00
2013-09-10 23:08:35 +00:00
gswe_moment_calculate_planet ( moment , planet_data - > planet_id , NULL ) ;
2013-09-02 21:10:35 +00:00
point = GPOINTER_TO_INT ( g_hash_table_lookup ( moment - > priv - > element_points , GINT_TO_POINTER ( planet_data - > sign - > element ) ) ) + planet_data - > planet_info - > points ;
g_hash_table_replace ( moment - > priv - > element_points , GINT_TO_POINTER ( planet_data - > sign - > element ) , GINT_TO_POINTER ( point ) ) ;
point = GPOINTER_TO_INT ( g_hash_table_lookup ( moment - > priv - > quality_points , GINT_TO_POINTER ( planet_data - > sign - > quality ) ) ) ;
2013-09-02 14:19:45 +00:00
2013-09-02 21:10:35 +00:00
point + = planet_data - > planet_info - > points ;
g_hash_table_replace ( moment - > priv - > quality_points , GINT_TO_POINTER ( planet_data - > sign - > quality ) , GINT_TO_POINTER ( point ) ) ;
}
static void
gswe_moment_calculate_points ( GsweMoment * moment )
{
if ( moment - > priv - > points_revision = = moment - > priv - > revision ) {
2013-09-02 14:19:45 +00:00
return ;
}
2013-09-02 21:10:35 +00:00
g_hash_table_remove_all ( moment - > priv - > element_points ) ;
g_hash_table_remove_all ( moment - > priv - > quality_points ) ;
2013-09-02 14:19:45 +00:00
2013-09-02 21:10:35 +00:00
g_list_foreach ( moment - > priv - > planet_list , ( GFunc ) add_points , moment ) ;
moment - > priv - > points_revision = moment - > priv - > revision ;
2013-09-02 14:19:45 +00:00
}
2013-09-02 21:10:35 +00:00
guint
gswe_moment_get_element_points ( GsweMoment * moment , GsweElement element )
2013-09-02 14:19:45 +00:00
{
2013-09-02 21:10:35 +00:00
guint point ;
2013-09-02 14:19:45 +00:00
2013-09-02 21:10:35 +00:00
gswe_moment_calculate_points ( moment ) ;
2013-09-02 14:19:45 +00:00
2013-09-02 21:10:35 +00:00
point = GPOINTER_TO_INT ( g_hash_table_lookup ( moment - > priv - > element_points , GINT_TO_POINTER ( element ) ) ) ;
2013-09-02 14:19:45 +00:00
2013-09-02 21:10:35 +00:00
return point ;
}
guint
gswe_moment_get_quality_points ( GsweMoment * moment , GsweQuality quality )
{
guint point ;
gswe_moment_calculate_points ( moment ) ;
point = GPOINTER_TO_INT ( g_hash_table_lookup ( moment - > priv - > quality_points , GINT_TO_POINTER ( quality ) ) ) ;
return point ;
2013-09-02 14:19:45 +00:00
}
2013-09-02 21:12:56 +00:00
GsweMoonPhaseData *
2013-09-10 23:08:35 +00:00
gswe_moment_get_moon_phase ( GsweMoment * moment , GError * * err )
2013-09-02 21:12:56 +00:00
{
gdouble difference ,
2013-09-10 23:08:35 +00:00
phase_percent ,
jd ,
jdb ;
2013-09-02 21:12:56 +00:00
if ( moment - > priv - > moon_phase_revision = = moment - > priv - > revision ) {
return & ( moment - > priv - > moon_phase ) ;
}
2013-09-10 23:08:35 +00:00
jd = gswe_timestamp_get_julian_day ( moment - > priv - > timestamp , err ) ;
if ( ( err ) & & ( * err ) ) {
return NULL ;
}
jdb = gswe_timestamp_get_julian_day ( gswe_full_moon_base_date , err ) ;
if ( ( err ) & & ( * err ) ) {
return NULL ;
}
difference = ( jd - jdb ) ;
2013-09-02 21:12:56 +00:00
phase_percent = fmod ( ( difference * 100 ) / SYNODIC , 100 ) ;
if ( phase_percent < 0 ) {
phase_percent + = 100.0 ;
}
if ( ( phase_percent < 0 ) | | ( phase_percent > 100 ) ) {
g_error ( " Error during Moon phase calculation! " ) ;
}
moment - > priv - > moon_phase . illumination = ( 50.0 - fabs ( phase_percent - 50.0 ) ) * 2 ;
if ( phase_percent = = 0 ) {
moment - > priv - > moon_phase . phase = GSWE_MOON_PHASE_NEW ;
} else if ( phase_percent < 25 ) {
moment - > priv - > moon_phase . phase = GSWE_MOON_PHASE_WAXING_CRESCENT ;
} else if ( phase_percent = = 25 ) {
moment - > priv - > moon_phase . phase = GSWE_MOON_PHASE_WAXING_HALF ;
} else if ( phase_percent < 50 ) {
moment - > priv - > moon_phase . phase = GSWE_MOON_PHASE_WAXING_GIBBOUS ;
} else if ( phase_percent = = 50 ) {
moment - > priv - > moon_phase . phase = GSWE_MOON_PHASE_FULL ;
} else if ( phase_percent < 75 ) {
moment - > priv - > moon_phase . phase = GSWE_MOON_PHASE_WANING_GIBBOUS ;
} else if ( phase_percent = = 75 ) {
moment - > priv - > moon_phase . phase = GSWE_MOON_PHASE_WANING_HALF ;
} else if ( phase_percent < 100 ) {
moment - > priv - > moon_phase . phase = GSWE_MOON_PHASE_WANING_CRESCENT ;
} else {
moment - > priv - > moon_phase . phase = GSWE_MOON_PHASE_DARK ;
}
moment - > priv - > moon_phase_revision = moment - > priv - > revision ;
return & ( moment - > priv - > moon_phase ) ;
}
2013-09-02 23:11:18 +00:00
static gint
find_aspect_by_both_planets ( GsweAspectData * aspect , struct GsweAspectFinder * aspect_finder )
{
if ( ( ( aspect - > planet1 - > planet_id = = aspect_finder - > planet1 ) & & ( aspect - > planet2 - > planet_id = = aspect_finder - > planet2 ) ) | | ( ( aspect - > planet1 - > planet_id = = aspect_finder - > planet2 ) & & ( aspect - > planet2 - > planet_id = = aspect_finder - > planet1 ) ) ) {
return 0 ;
}
return 1 ;
}
static gboolean
find_aspect ( gpointer aspect_p , GsweAspectInfo * aspect_info , GsweAspectData * aspect_data )
{
GsweAspect aspect = GPOINTER_TO_INT ( aspect_p ) ;
2013-09-03 09:26:09 +00:00
gdouble diff ,
2013-09-02 23:11:18 +00:00
planet_orb ,
aspect_orb ;
2013-09-03 09:26:09 +00:00
aspect_data - > distance = fabs ( aspect_data - > planet1 - > position - aspect_data - > planet2 - > position ) ;
2013-09-02 23:11:18 +00:00
2013-09-03 09:26:09 +00:00
if ( aspect_data - > distance > 180.0 ) {
aspect_data - > distance = 360.0 - aspect_data - > distance ;
2013-09-02 23:11:18 +00:00
}
2013-09-03 09:26:09 +00:00
diff = fabs ( aspect_info - > size - aspect_data - > distance ) ;
2013-09-02 23:11:18 +00:00
planet_orb = fmin ( aspect_data - > planet1 - > planet_info - > orb , aspect_data - > planet2 - > planet_info - > orb ) ;
aspect_orb = fmax ( 1.0 , planet_orb - aspect_info - > orb_modifier ) ;
if ( diff < aspect_orb ) {
aspect_data - > aspect = aspect ;
aspect_data - > aspect_info = aspect_info ;
if ( aspect_info - > size = = 0 ) {
aspect_data - > difference = ( 1 - ( ( 360.0 - diff ) / 360.0 ) ) * 100.0 ;
} else {
aspect_data - > difference = ( 1 - ( ( aspect_info - > size - diff ) / aspect_info - > size ) ) * 100.0 ;
}
return TRUE ;
}
return FALSE ;
}
static void
gswe_moment_calculate_aspects ( GsweMoment * moment )
{
GList * oplanet ,
* iplanet ;
if ( moment - > priv - > aspect_revision = = moment - > priv - > revision ) {
return ;
}
gswe_moment_calculate_all_planets ( moment ) ;
g_list_free_full ( moment - > priv - > aspect_list , g_free ) ;
for ( oplanet = moment - > priv - > planet_list ; oplanet ; oplanet = oplanet - > next ) {
for ( iplanet = moment - > priv - > planet_list ; iplanet ; iplanet = iplanet - > next ) {
GswePlanetData * outer_planet = oplanet - > data ,
* inner_planet = iplanet - > data ;
struct GsweAspectFinder aspect_finder ;
GsweAspectData * aspect_data ;
if ( outer_planet - > planet_id = = inner_planet - > planet_id ) {
continue ;
}
aspect_finder . planet1 = outer_planet - > planet_id ;
aspect_finder . planet2 = inner_planet - > planet_id ;
if ( g_list_find_custom ( moment - > priv - > aspect_list , & aspect_finder , ( GCompareFunc ) find_aspect_by_both_planets ) ! = NULL ) {
continue ;
}
aspect_data = g_new0 ( GsweAspectData , 1 ) ;
aspect_data - > planet1 = outer_planet ;
aspect_data - > planet2 = inner_planet ;
aspect_data - > aspect = GSWE_ASPECT_NONE ;
( void ) g_hash_table_find ( gswe_aspect_info_table , ( GHRFunc ) find_aspect , aspect_data ) ;
if ( aspect_data - > aspect = = GSWE_ASPECT_NONE ) {
aspect_data - > aspect_info = g_hash_table_lookup ( gswe_aspect_info_table , GINT_TO_POINTER ( GSWE_ASPECT_NONE ) ) ;
}
moment - > priv - > aspect_list = g_list_prepend ( moment - > priv - > aspect_list , aspect_data ) ;
}
}
moment - > priv - > aspect_revision = moment - > priv - > revision ;
}
2013-09-03 09:26:53 +00:00
/**
2013-09-10 22:17:23 +00:00
* gswe_moment_get_all_aspects :
2013-09-03 09:26:53 +00:00
* @ moment : the GsweMoment to operate on
*
* Gets all planetary aspects between the planets added by
* gswe_moment_add_planet ( ) or gswe_moment_add_all_planets ( ) .
*
2013-09-05 17:50:57 +00:00
* Returns : ( element - type GsweAspectData ) ( transfer none ) : a GList of
* # GsweAspectData . Both the GList and GsweAspectData objects belong
* to @ moment , and should not be freed or modified .
2013-09-03 09:26:53 +00:00
*/
2013-09-02 23:11:18 +00:00
GList *
2013-09-10 22:17:23 +00:00
gswe_moment_get_all_aspects ( GsweMoment * moment )
2013-09-02 23:11:18 +00:00
{
gswe_moment_calculate_aspects ( moment ) ;
return moment - > priv - > aspect_list ;
}
2013-09-03 09:26:53 +00:00
/**
* gswe_moment_get_planet_aspects :
* @ moment : the GsweMoment to operate on
* @ planet : the planet whose aspects you want to get
2013-09-10 23:55:10 +00:00
* @ err : a # GError
2013-09-03 09:26:53 +00:00
*
* Get all the aspects between @ planet and all the other planets added with
* gswe_moment_add_planet ( ) or gswe_moment_add_all_planets ( ) .
*
2013-09-05 17:50:57 +00:00
* Returns : ( element - type GsweAspectData ) ( transfer container ) : a # GList of
* # GsweAspectData . The GsweAspectData structures belong to @ moment ,
* but the GList should be freed using g_list_free ( ) . If the planet
* has no aspects , or the planet has not been added to @ moment ,
2013-09-03 09:26:53 +00:00
* returns NULL .
*/
2013-09-02 23:11:18 +00:00
GList *
2013-09-10 23:08:35 +00:00
gswe_moment_get_planet_aspects ( GsweMoment * moment , GswePlanet planet , GError * * err )
2013-09-02 23:11:18 +00:00
{
2013-09-03 09:26:37 +00:00
GList * ret = NULL ,
* aspect ;
2013-09-10 23:08:35 +00:00
if ( ! gswe_moment_has_planet ( moment , planet ) ) {
g_set_error ( err , GSWE_MOMENT_ERROR , GSWE_MOMENT_ERROR_NONADDED_PLANET , " Specified planet is not added to the moment object " ) ;
return NULL ;
}
2013-09-03 09:26:37 +00:00
gswe_moment_calculate_aspects ( moment ) ;
for ( aspect = moment - > priv - > aspect_list ; aspect ; aspect = aspect - > next ) {
GsweAspectData * aspect_data = aspect - > data ;
if ( ( aspect_data - > planet1 - > planet_id = = planet ) | | ( aspect_data - > planet2 - > planet_id = = planet ) ) {
ret = g_list_prepend ( ret , aspect_data ) ;
}
}
return ret ;
2013-09-02 23:11:18 +00:00
}
2013-09-03 10:59:32 +00:00
static gboolean
find_mirror ( gpointer mirror_p , GsweMirrorInfo * mirror_info , GsweMirrorData * mirror_data )
{
GsweMirror mirror = GPOINTER_TO_INT ( mirror_p ) ;
gdouble start_point ,
mirror_position ,
planet_orb ;
if ( mirror = = GSWE_MIRROR_NONE ) {
return FALSE ;
}
planet_orb = fmin ( mirror_data - > planet1 - > planet_info - > orb , mirror_data - > planet2 - > planet_info - > orb ) ;
start_point = ( mirror_info - > start_sign - > sign_id - 1 ) * 30.0 ;
if ( mirror_info - > middle_axis = = TRUE ) {
start_point + = 15.0 ;
}
mirror_position = 2 * start_point - mirror_data - > planet1 - > position ;
if ( mirror_position < 0 ) {
mirror_position + = 360.0 ;
}
if ( ( mirror_data - > difference = fabs ( mirror_data - > planet2 - > position - mirror_position ) ) < = planet_orb ) {
mirror_data - > mirror_info = mirror_info ;
mirror_data - > mirror = mirror ;
return TRUE ;
} else {
mirror_data - > difference = 0.0 ;
}
return FALSE ;
}
static gint
find_mirror_by_both_planets ( GsweMirrorData * mirror , struct GsweAspectFinder * mirror_finder )
{
if ( ( ( mirror - > planet1 - > planet_id = = mirror_finder - > planet1 ) & & ( mirror - > planet2 - > planet_id = = mirror_finder - > planet2 ) ) | | ( ( mirror - > planet1 - > planet_id = = mirror_finder - > planet2 ) & & ( mirror - > planet2 - > planet_id = = mirror_finder - > planet1 ) ) ) {
return 0 ;
}
return 1 ;
}
static void
gswe_moment_calculate_mirrorpoints ( GsweMoment * moment )
{
GList * oplanet ,
* iplanet ;
if ( moment - > priv - > mirrorpoint_revision = = moment - > priv - > revision ) {
return ;
}
gswe_moment_calculate_all_planets ( moment ) ;
g_list_free_full ( moment - > priv - > mirrorpoint_list , g_free ) ;
for ( oplanet = moment - > priv - > planet_list ; oplanet ; oplanet = oplanet - > next ) {
for ( iplanet = moment - > priv - > planet_list ; iplanet ; iplanet = iplanet - > next ) {
GswePlanetData * outer_planet = oplanet - > data ,
* inner_planet = iplanet - > data ;
GsweMirrorData * mirror_data ;
struct GsweAspectFinder mirror_finder ;
if ( outer_planet - > planet_id = = inner_planet - > planet_id ) {
continue ;
}
mirror_finder . planet1 = outer_planet - > planet_id ;
mirror_finder . planet2 = inner_planet - > planet_id ;
if ( g_list_find_custom ( moment - > priv - > mirrorpoint_list , & mirror_finder , ( GCompareFunc ) find_mirror_by_both_planets ) ! = NULL ) {
continue ;
}
mirror_data = g_new0 ( GsweMirrorData , 1 ) ;
mirror_data - > planet1 = outer_planet ;
mirror_data - > planet2 = inner_planet ;
mirror_data - > mirror = GSWE_MIRROR_NONE ;
( void ) g_hash_table_find ( gswe_mirror_info_table , ( GHRFunc ) find_mirror , mirror_data ) ;
if ( mirror_data - > mirror = = GSWE_MIRROR_NONE ) {
mirror_data - > mirror_info = g_hash_table_lookup ( gswe_mirror_info_table , GINT_TO_POINTER ( GSWE_MIRROR_NONE ) ) ;
}
moment - > priv - > mirrorpoint_list = g_list_prepend ( moment - > priv - > mirrorpoint_list , mirror_data ) ;
}
}
moment - > priv - > mirrorpoint_revision = moment - > priv - > revision ;
}
2013-09-05 17:50:57 +00:00
/**
* gswe_moment_get_all_mirrorpoints :
* @ moment : The GsweMoment object to operate on .
*
* Get all found mirrorpoints between planets in @ moment .
*
* Returns : ( element - type GsweMirrorData ) ( transfer none ) : A # GList of
* # GsweMirrorData .
*/
2013-09-03 10:59:32 +00:00
GList *
gswe_moment_get_all_mirrorpoints ( GsweMoment * moment )
{
gswe_moment_calculate_mirrorpoints ( moment ) ;
return moment - > priv - > mirrorpoint_list ;
}
2013-09-05 17:50:57 +00:00
/**
* gswe_moment_get_all_planet_mirrorpoints :
* @ moment : The GsweMoment object to operate on .
* @ planet : The planet whose mirrorpoint planets are requested .
2013-09-10 23:55:10 +00:00
* @ err : a # GError
2013-09-05 17:50:57 +00:00
*
* Get all the mirrorpoint planets on all registered mirrors for @ planet .
*
* Returns : ( element - type GsweMirrorData ) ( transfer container ) : a # GList of
* # GsweMirrorData . The GsweMirrorData structures belong to @ moment ,
* but the GList should be freed using g_list_free ( ) . If no planet
* has any mirrorpoints , or the planet has not been added to @ moment ,
* returns NULL .
*/
2013-09-03 10:59:32 +00:00
GList *
2013-09-10 23:08:35 +00:00
gswe_moment_get_all_planet_mirrorpoints ( GsweMoment * moment , GswePlanet planet , GError * * err )
2013-09-03 10:59:32 +00:00
{
GList * ret = NULL ,
* mirror ;
2013-09-10 23:08:35 +00:00
if ( ! gswe_moment_has_planet ( moment , planet ) ) {
g_set_error ( err , GSWE_MOMENT_ERROR , GSWE_MOMENT_ERROR_NONADDED_PLANET , " Specified planet is not added to the moment object " ) ;
return NULL ;
}
2013-09-03 10:59:32 +00:00
gswe_moment_calculate_mirrorpoints ( moment ) ;
for ( mirror = moment - > priv - > mirrorpoint_list ; mirror ; mirror = mirror - > next ) {
GsweMirrorData * mirror_data = mirror - > data ;
if ( ( mirror_data - > planet1 - > planet_id = = planet ) | | ( mirror_data - > planet2 - > planet_id = = planet ) ) {
ret = g_list_prepend ( ret , mirror_data ) ;
}
}
return ret ;
}
2013-09-05 17:50:57 +00:00
/**
2013-09-10 22:18:31 +00:00
* gswe_moment_get_mirror_all_mirrorpoints :
2013-09-05 17:50:57 +00:00
* @ moment : The GsweMoment object to operate on .
* @ mirror : The mirror on which you want to search for mirrored planets .
*
* Get all the mirrorpoint planets on the specified mirror @ mirror .
*
* Returns : ( element - type GsweMirrorData ) ( transfer container ) : a # GList of
* # GsweMirrorData . The GsweMirrorData structures belong to @ moment ,
* but the GList should be freed using g_list_free ( ) . If there are
* no mirrored planets on the given mirror , returns NULL .
*/
2013-09-03 10:59:32 +00:00
GList *
2013-09-10 22:18:31 +00:00
gswe_moment_get_mirror_all_mirrorpoints ( GsweMoment * moment , GsweMirror mirror )
2013-09-03 10:59:32 +00:00
{
GList * ret = NULL ,
* mirror_l ;
gswe_moment_calculate_mirrorpoints ( moment ) ;
for ( mirror_l = moment - > priv - > mirrorpoint_list ; mirror_l ; mirror_l = mirror_l - > next ) {
GsweMirrorData * mirror_data = mirror_l - > data ;
if ( mirror_data - > mirror = = mirror ) {
ret = g_list_prepend ( ret , mirror_data ) ;
}
}
return ret ;
}
2013-09-05 17:50:57 +00:00
/**
* gswe_moment_get_mirror_planet_mirrorpoints :
* @ moment : the GsweMoment object to operate on
* @ mirror : the mirror on which you want to search for mirrored planets
* @ planet : the planet whose mirrorpoint planets are requested
2013-09-10 23:55:10 +00:00
* @ err : a # GError
2013-09-05 17:50:57 +00:00
*
* Get the mirrorpoint planets of @ planet as seen in @ mirror .
*
* Returns : ( element - type GsweMirrorData ) ( transfer container ) : a # GList of
* # GsweMirrorData . The GsweMirrorData structires belong to @ moment ,
* but the GList should be freed using g_list_free ( ) . If the planet
* has no mirrorpoints , or the planet has not been added to @ moment ,
* returns NULL .
*/
2013-09-03 10:59:32 +00:00
GList *
2013-09-10 23:08:35 +00:00
gswe_moment_get_mirror_planet_mirrorpoints ( GsweMoment * moment , GsweMirror mirror , GswePlanet planet , GError * * err )
2013-09-03 10:59:32 +00:00
{
GList * ret = NULL ,
* mirror_l ;
2013-09-10 23:08:35 +00:00
if ( ! gswe_moment_has_planet ( moment , planet ) ) {
g_set_error ( err , GSWE_MOMENT_ERROR , GSWE_MOMENT_ERROR_NONADDED_PLANET , " Specified planet is not added to the moment object " ) ;
return NULL ;
}
2013-09-03 10:59:32 +00:00
gswe_moment_calculate_mirrorpoints ( moment ) ;
for ( mirror_l = moment - > priv - > mirrorpoint_list ; mirror_l ; mirror_l = mirror_l - > next ) {
GsweMirrorData * mirror_data = mirror_l - > data ;
if ( ( ( mirror_data - > planet1 - > planet_id = = planet ) | | ( mirror_data - > planet2 - > planet_id = = planet ) ) & & ( mirror_data - > mirror = = mirror ) ) {
ret = g_list_prepend ( ret , mirror_data ) ;
}
}
return ret ;
}
2013-09-06 07:32:52 +00:00
static GsweMoonPhaseData *
gswe_moon_phase_data_copy ( GsweMoonPhaseData * moon_phase_data )
{
GsweMoonPhaseData * ret = g_new0 ( GsweMoonPhaseData , 1 ) ;
ret - > phase = moon_phase_data - > phase ;
ret - > illumination = moon_phase_data - > illumination ;
return ret ;
}
2013-09-06 07:36:09 +00:00
/**
* gswe_moon_phase_data_get_type : ( skip )
*
* Register the # GsweMoonPhaseData struct as a # GBoxedType . It is required for
* GObject Introspection . You should never need to call this directly .
*
* Returns : the newly registered type ID
*/
2013-09-06 07:32:52 +00:00
GType
gswe_moon_phase_data_get_type ( void )
{
2013-09-11 01:31:35 +00:00
return g_boxed_type_register_static ( " GsweMoonPhaseData " , ( GBoxedCopyFunc ) gswe_moon_phase_data_copy , ( GBoxedFreeFunc ) g_free ) ;
2013-09-06 07:32:52 +00:00
}
2013-09-11 01:31:35 +00:00
static GswePlanetData *
2013-09-06 07:32:52 +00:00
gswe_planet_data_copy ( GswePlanetData * planet_data )
{
GswePlanetData * ret = g_new0 ( GswePlanetData , 1 ) ;
ret - > planet_id = planet_data - > planet_id ;
ret - > planet_info = planet_data - > planet_info ;
ret - > position = planet_data - > position ;
ret - > retrograde = planet_data - > retrograde ;
ret - > house = planet_data - > house ;
ret - > sign = planet_data - > sign ;
ret - > revision = planet_data - > revision ;
return ret ;
}
2013-09-06 07:36:09 +00:00
/**
* gswe_planet_data_get_type : ( skip )
*
* Register the # GswePlanetData struct as a # GBoxedType . It is required for GObject Introspection . You should never need to call this directly .
*
* Returns : the newly registered type ID
*/
2013-09-06 07:32:52 +00:00
GType
gswe_planet_data_get_type ( void )
{
2013-09-11 01:31:35 +00:00
return g_boxed_type_register_static ( " GswePlanetData " , ( GBoxedCopyFunc ) gswe_planet_data_copy , ( GBoxedFreeFunc ) g_free ) ;
2013-09-06 07:32:52 +00:00
}
2013-09-11 01:32:20 +00:00
static GsweCoordinates *
gswe_coordinates_copy ( GsweCoordinates * coordinates )
{
GsweCoordinates * ret = g_new0 ( GsweCoordinates , 1 ) ;
ret - > longitude = coordinates - > longitude ;
ret - > latitude = coordinates - > latitude ;
ret - > altitude = coordinates - > altitude ;
return ret ;
}
/**
* gswe_coordinates_get_type :
*
* Register the # Gswecoordinates struct as a # GBoxedType . It is required for GObject Introspection . You should never need to call this directly .
*
* Returns : the newly registered type ID
*/
GType
gswe_coordinates_get_type ( void )
{
return g_boxed_type_register_static ( " GsweCoordinates " , ( GBoxedCopyFunc ) gswe_coordinates_copy , ( GBoxedFreeFunc ) g_free ) ;
}