#include #include #include #include #include #include #include #include #include #include #include "config.h" #include "ag-app.h" #include "ag-window.h" #include "ag-chart.h" #include "ag-settings.h" #include "ag-db.h" struct _AgWindowPrivate { GtkWidget *header_bar; GtkWidget *menubutton_stack; GtkWidget *new_back_stack; GtkWidget *selection_toolbar; GtkWidget *stack; GtkWidget *name; GtkWidget *country; GtkWidget *city; GtkWidget *north_lat; GtkWidget *south_lat; GtkWidget *latitude; GtkWidget *east_long; GtkWidget *west_long; GtkWidget *longitude; GtkWidget *year; GtkWidget *month; GtkWidget *day; GtkWidget *hour; GtkWidget *minute; GtkWidget *second; GtkWidget *timezone; GtkWidget *house_system; GtkWidget *tab_list; GtkWidget *tab_chart; GtkWidget *tab_edit; GtkWidget *current_tab; GtkWidget *aspect_table; GtkWidget *chart_web_view; GtkAdjustment *year_adjust; AgSettings *settings; AgChart *chart; gboolean aspect_table_populated; GtkTextBuffer *note_buffer; GtkListStore *house_system_model; GtkListStore *db_chart_data; AgDbSave *saved_data; GtkEntryCompletion *country_comp; GtkEntryCompletion *city_comp; gchar *selected_country; gchar *selected_city; }; struct cc_search { const gchar *target; GtkTreeIter *ret_iter; gchar *ret_code; }; G_DEFINE_QUARK(ag_window_error_quark, ag_window_error); G_DEFINE_TYPE_WITH_PRIVATE(AgWindow, ag_window, GTK_TYPE_APPLICATION_WINDOW); static void ag_window_gear_menu_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { GVariant *state; state = g_action_get_state(G_ACTION(action)); g_action_change_state( G_ACTION(action), g_variant_new_boolean(!g_variant_get_boolean(state)) ); g_variant_unref(state); } static void ag_window_view_menu_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { GVariant *state; state = g_action_get_state(G_ACTION(action)); g_action_change_state( G_ACTION(action), g_variant_new_boolean(!g_variant_get_boolean(state)) ); g_variant_unref(state); } const gchar * ag_window_planet_character(GswePlanet planet) { switch (planet) { case GSWE_PLANET_ASCENDANT: return "AC"; case GSWE_PLANET_MC: return "MC"; case GSWE_PLANET_VERTEX: return "Vx"; case GSWE_PLANET_SUN: return "☉"; case GSWE_PLANET_MOON: return "☽"; case GSWE_PLANET_MOON_NODE: return "☊"; case GSWE_PLANET_MERCURY: return "☿"; case GSWE_PLANET_VENUS: return "♀"; case GSWE_PLANET_MARS: return "♂"; case GSWE_PLANET_JUPITER: return "♃"; case GSWE_PLANET_SATURN: return "♄"; case GSWE_PLANET_URANUS: return "♅"; case GSWE_PLANET_NEPTUNE: return "♆"; case GSWE_PLANET_PLUTO: return "♇"; case GSWE_PLANET_CERES: return "⚳"; case GSWE_PLANET_PALLAS: return "⚴"; case GSWE_PLANET_JUNO: return "⚵"; case GSWE_PLANET_VESTA: return "⚶"; case GSWE_PLANET_CHIRON: return "⚷"; case GSWE_PLANET_MOON_APOGEE: return "⚸"; default: return NULL; } } GtkWidget * ag_window_create_planet_widget(GswePlanetInfo *planet_info) { const gchar *planet_char; GswePlanet planet = gswe_planet_info_get_planet(planet_info); GSettings *settings = ag_settings_peek_main_settings(ag_settings_get()); if ( ((planet_char = ag_window_planet_character(planet)) != NULL) && (g_settings_get_boolean(settings, "planets-char")) ) { return gtk_label_new(planet_char); } switch (planet) { case GSWE_PLANET_SUN: return gtk_image_new_from_resource( "/eu/polonkai/gergely" "/Astrognome/default-icons/planet-sun.svg" ); default: return gtk_label_new(gswe_planet_info_get_name(planet_info)); } } const gchar * ag_window_aspect_character(GsweAspect aspect) { switch (aspect) { case GSWE_ASPECT_CONJUCTION: return "☌"; case GSWE_ASPECT_OPPOSITION: return "☍"; case GSWE_ASPECT_QUINTILE: return "Q"; case GSWE_ASPECT_BIQUINTILE: return "BQ"; case GSWE_ASPECT_SQUARE: return "◽"; case GSWE_ASPECT_TRINE: return "▵"; case GSWE_ASPECT_SEXTILE: return "⚹"; case GSWE_ASPECT_SEMISEXTILE: return "⚺"; case GSWE_ASPECT_QUINCUNX: return "⚻"; case GSWE_ASPECT_SESQUISQUARE: return "⚼"; default: return NULL; } } GtkWidget * ag_window_create_aspect_widget(GsweAspectInfo *aspect_info) { const gchar *aspect_char; GsweAspect aspect = gswe_aspect_info_get_aspect(aspect_info); GSettings *settings = ag_settings_peek_main_settings(ag_settings_get()); if ( ((aspect_char = ag_window_aspect_character(aspect)) != NULL) && (g_settings_get_boolean(settings, "aspects-char")) ) { return gtk_label_new(aspect_char); } switch (aspect) { default: return gtk_label_new(gswe_aspect_info_get_name(aspect_info)); } } void ag_window_redraw_aspect_table(AgWindow *window) { GList *planet_list, *planet1, *planet2; guint i, j; AgWindowPrivate *priv = ag_window_get_instance_private(window); planet_list = ag_chart_get_planets(priv->chart); if (priv->aspect_table_populated == FALSE) { GList *planet; guint i; for ( planet = planet_list, i = 0; planet; planet = g_list_next(planet), i++ ) { GtkWidget *label_hor, *label_ver, *current_widget; GswePlanet planet_id; GswePlanetData *planet_data; GswePlanetInfo *planet_info; planet_id = GPOINTER_TO_INT(planet->data); planet_data = gswe_moment_get_planet( GSWE_MOMENT(priv->chart), planet_id, NULL ); planet_info = gswe_planet_data_get_planet_info(planet_data); if ((current_widget = gtk_grid_get_child_at( GTK_GRID(priv->aspect_table), i + 1, i )) != NULL) { gtk_container_remove( GTK_CONTAINER(priv->aspect_table), current_widget ); } label_hor = ag_window_create_planet_widget(planet_info); gtk_grid_attach( GTK_GRID(priv->aspect_table), label_hor, i + 1, i, 1, 1 ); if (i > 0) { if ((current_widget = gtk_grid_get_child_at( GTK_GRID(priv->aspect_table), 0, i )) != NULL) { gtk_container_remove( GTK_CONTAINER(priv->aspect_table), current_widget ); } label_ver = ag_window_create_planet_widget(planet_info); gtk_grid_attach( GTK_GRID(priv->aspect_table), label_ver, 0, i, 1, 1 ); } } priv->aspect_table_populated = TRUE; } for ( planet1 = planet_list, i = 0; planet1; planet1 = g_list_next(planet1), i++ ) { for ( planet2 = planet_list, j = 0; planet2; planet2 = g_list_next(planet2), j++ ) { GsweAspectData *aspect; GtkWidget *aspect_widget; GError *err = NULL; if (GPOINTER_TO_INT(planet1->data) == GPOINTER_TO_INT(planet2->data) ) { break; } if ((aspect_widget = gtk_grid_get_child_at( GTK_GRID(priv->aspect_table), j + 1, i )) != NULL) { gtk_container_remove( GTK_CONTAINER(priv->aspect_table), aspect_widget ); } if ((aspect = gswe_moment_get_aspect_by_planets( GSWE_MOMENT(priv->chart), GPOINTER_TO_INT(planet1->data), GPOINTER_TO_INT(planet2->data), &err )) != NULL) { GsweAspectInfo *aspect_info; aspect_info = gswe_aspect_data_get_aspect_info(aspect); if (gswe_aspect_data_get_aspect(aspect) != GSWE_ASPECT_NONE) { aspect_widget = ag_window_create_aspect_widget(aspect_info); gtk_grid_attach( GTK_GRID(priv->aspect_table), aspect_widget, j + 1, i, 1, 1 ); } } else if (err) { g_warning("%s\n", err->message); } else { g_error( "No aspect is returned between two planets. " \ "This is a bug in SWE-GLib!" ); } } } gtk_widget_show_all(priv->aspect_table); } /** * ag_window_redraw_chart: * @window: the #AgWindow to operate on * * Redraw the chart on the chart view. */ void ag_window_redraw_chart(AgWindow *window) { gsize length; GError *err = NULL; AgWindowPrivate *priv = ag_window_get_instance_private(window); gchar *svg_content = ag_chart_create_svg( priv->chart, &length, &err ); if (svg_content == NULL) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_WARNING, "Unable to draw chart: %s", err->message ); } else { GBytes *content; content = g_bytes_new_take(svg_content, length); webkit_web_view_load_bytes( WEBKIT_WEB_VIEW(priv->chart_web_view), content, "image/svg+xml", "UTF-8", NULL ); g_bytes_unref(content); } ag_window_redraw_aspect_table(window); } static gboolean ag_window_set_house_system(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, AgWindow *window) { GsweHouseSystem row_house_system; AgWindowPrivate *priv = ag_window_get_instance_private(window); GsweHouseSystem house_system = gswe_moment_get_house_system( GSWE_MOMENT(priv->chart) ); gtk_tree_model_get( GTK_TREE_MODEL(priv->house_system_model), iter, 0, &row_house_system, -1 ); if (house_system == row_house_system) { gtk_combo_box_set_active_iter(GTK_COMBO_BOX(priv->house_system), iter); return TRUE; } return FALSE; } void ag_window_update_from_chart(AgWindow *window) { AgWindowPrivate *priv = ag_window_get_instance_private(window); GsweTimestamp *timestamp = gswe_moment_get_timestamp( GSWE_MOMENT(priv->chart) ); GsweCoordinates *coordinates = gswe_moment_get_coordinates( GSWE_MOMENT(priv->chart) ); gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->year), gswe_timestamp_get_gregorian_year(timestamp, NULL) ); gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->month), gswe_timestamp_get_gregorian_month(timestamp, NULL) ); gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->day), gswe_timestamp_get_gregorian_day(timestamp, NULL) ); gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->hour), gswe_timestamp_get_gregorian_hour(timestamp, NULL) ); gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->minute), gswe_timestamp_get_gregorian_minute(timestamp, NULL) ); gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->second), gswe_timestamp_get_gregorian_second(timestamp, NULL)); gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->timezone), gswe_timestamp_get_gregorian_timezone(timestamp) ); gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->longitude), fabs(coordinates->longitude) ); if (coordinates->longitude < 0.0) { gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(priv->west_long), TRUE); } gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->latitude), fabs(coordinates->latitude) ); if (coordinates->latitude < 0.0) { gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(priv->south_lat), TRUE); } gtk_tree_model_foreach( GTK_TREE_MODEL(priv->house_system_model), (GtkTreeModelForeachFunc)ag_window_set_house_system, window ); gtk_entry_set_text(GTK_ENTRY(priv->name), ag_chart_get_name(priv->chart)); gtk_entry_set_text( GTK_ENTRY(priv->country), ag_chart_get_country(priv->chart) ); gtk_entry_set_text(GTK_ENTRY(priv->city), ag_chart_get_city(priv->chart)); if (ag_chart_get_note(priv->chart)) { // TODO: maybe setting length to -1 here is not that good of an idea… gtk_text_buffer_set_text( GTK_TEXT_BUFFER(priv->note_buffer), ag_chart_get_note(priv->chart), -1 ); } gtk_header_bar_set_subtitle( GTK_HEADER_BAR(priv->header_bar), ag_chart_get_name(priv->chart) ); g_free(coordinates); ag_window_redraw_chart(window); } static void ag_window_chart_changed(AgChart *chart, AgWindow *window) { ag_window_redraw_chart(window); } static void ag_window_recalculate_chart(AgWindow *window, gboolean set_everything) { AgDbSave *edit_data, *chart_data; AgWindowPrivate *priv = ag_window_get_instance_private(window); gboolean south, west; GtkTreeIter house_system_iter; GsweHouseSystem house_system; GtkTextIter start_iter, end_iter; GsweTimestamp *timestamp; GtkWidget *current; gint db_id = (priv->saved_data) ? priv->saved_data->db_id : -1; south = gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON(priv->south_lat) ); west = gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON(priv->west_long) ); // If the current widget is a spin button, force it to update. This is // required when the user enters a new value in a spin button, but doesn't // leave the spin entry before switching to the chart tab with an accel. current = gtk_window_get_focus(GTK_WINDOW(window)); if (GTK_IS_SPIN_BUTTON(current)) { gtk_spin_button_update(GTK_SPIN_BUTTON(current)); } edit_data = g_new0(AgDbSave, 1); edit_data->db_id = db_id; edit_data->name = g_strdup(gtk_entry_get_text(GTK_ENTRY(priv->name))); edit_data->country = g_strdup(gtk_entry_get_text(GTK_ENTRY(priv->country))); edit_data->city = g_strdup(gtk_entry_get_text(GTK_ENTRY(priv->city))); edit_data->longitude = gtk_spin_button_get_value( GTK_SPIN_BUTTON(priv->longitude) ); if (west) { edit_data->longitude = - edit_data->longitude; } edit_data->latitude = gtk_spin_button_get_value( GTK_SPIN_BUTTON(priv->latitude) ); if (south) { edit_data->latitude = - edit_data->latitude; } // TODO: This will cause problems with imported charts edit_data->altitude = DEFAULT_ALTITUDE; edit_data->year = gtk_spin_button_get_value_as_int( GTK_SPIN_BUTTON(priv->year) ); edit_data->month = gtk_spin_button_get_value_as_int( GTK_SPIN_BUTTON(priv->month) ); edit_data->day = gtk_spin_button_get_value_as_int( GTK_SPIN_BUTTON(priv->day) ); edit_data->hour = gtk_spin_button_get_value_as_int( GTK_SPIN_BUTTON(priv->hour) ); edit_data->minute = gtk_spin_button_get_value_as_int( GTK_SPIN_BUTTON(priv->minute) ); edit_data->second = gtk_spin_button_get_value_as_int( GTK_SPIN_BUTTON(priv->second) ); edit_data->timezone = gtk_spin_button_get_value( GTK_SPIN_BUTTON(priv->timezone) ); if (!gtk_combo_box_get_active_iter( GTK_COMBO_BOX(priv->house_system), &house_system_iter )) { g_error("House system is not set! This is clearly a bug."); } gtk_tree_model_get( GTK_TREE_MODEL(priv->house_system_model), &house_system_iter, 0, &house_system, -1 ); edit_data->house_system = g_strdup( ag_house_system_id_to_nick(house_system) ); gtk_text_buffer_get_bounds(priv->note_buffer, &start_iter, &end_iter); edit_data->note = gtk_text_buffer_get_text( priv->note_buffer, &start_iter, &end_iter, TRUE ); chart_data = (priv->chart) ? ag_chart_get_db_save(priv->chart, db_id) : NULL ; if (ag_db_save_identical(edit_data, chart_data, !set_everything)) { g_debug("No redrawing needed"); ag_db_save_data_free(edit_data); ag_db_save_data_free(chart_data); return; } ag_db_save_data_free(chart_data); g_debug("Recalculating chart data"); // TODO: Set timezone according to the city selected! if (priv->chart == NULL) { timestamp = gswe_timestamp_new_from_gregorian_full( edit_data->year, edit_data->month, edit_data->day, edit_data->hour, edit_data->minute, edit_data->second, 0, edit_data->timezone ); priv->chart = ag_chart_new_full( timestamp, edit_data->longitude, edit_data->latitude, edit_data->altitude, house_system ); g_signal_connect( priv->chart, "changed", G_CALLBACK(ag_window_chart_changed), window ); ag_window_redraw_chart(window); } else { gswe_moment_set_house_system(GSWE_MOMENT(priv->chart), house_system); timestamp = gswe_moment_get_timestamp(GSWE_MOMENT(priv->chart)); gswe_timestamp_set_gregorian_full( timestamp, edit_data->year, edit_data->month, edit_data->day, edit_data->hour, edit_data->minute, edit_data->second, 0, edit_data->timezone, NULL ); } if (set_everything) { ag_chart_set_name(priv->chart, edit_data->name); ag_chart_set_country(priv->chart, edit_data->country); ag_chart_set_city(priv->chart, edit_data->city); ag_chart_set_note(priv->chart, edit_data->note); } ag_db_save_data_free(edit_data); } static void ag_window_export_svg(AgWindow *window, GError **err) { gchar *name; gchar *file_name; GtkWidget *fs; gint response; AgWindowPrivate *priv = ag_window_get_instance_private(window); ag_window_recalculate_chart(window, TRUE); // We should never enter here, but who knows... if (priv->chart == NULL) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, _("Chart cannot be calculated.") ); g_set_error( err, AG_WINDOW_ERROR, AG_WINDOW_ERROR_EMPTY_CHART, "Chart is empty" ); return; } name = ag_chart_get_name(priv->chart); if ((name == NULL) || (*name == 0)) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, _("You must enter a name before saving a chart.") ); g_set_error( err, AG_WINDOW_ERROR, AG_WINDOW_ERROR_NO_NAME, "No name specified" ); return; } file_name = g_strdup_printf("%s.svg", name); fs = gtk_file_chooser_dialog_new(_("Export Chart as SVG"), GTK_WINDOW(window), GTK_FILE_CHOOSER_ACTION_SAVE, _("_Cancel"), GTK_RESPONSE_CANCEL, _("_Save"), GTK_RESPONSE_ACCEPT, NULL); gtk_dialog_set_default_response(GTK_DIALOG(fs), GTK_RESPONSE_ACCEPT); gtk_file_chooser_set_local_only(GTK_FILE_CHOOSER(fs), FALSE); gtk_file_chooser_set_do_overwrite_confirmation(GTK_FILE_CHOOSER(fs), TRUE); gtk_file_chooser_set_current_name(GTK_FILE_CHOOSER(fs), file_name); g_free(file_name); response = gtk_dialog_run(GTK_DIALOG(fs)); gtk_widget_hide(fs); if (response == GTK_RESPONSE_ACCEPT) { GFile *file = gtk_file_chooser_get_file(GTK_FILE_CHOOSER(fs)); ag_chart_export_svg_to_file(priv->chart, file, err); } gtk_widget_destroy(fs); } static void ag_window_export_svg_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { AgWindow *window = AG_WINDOW(user_data); GError *err = NULL; ag_window_export_svg(window, &err); if (err) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, "%s", err->message ); } } static void ag_window_export(AgWindow *window, GError **err) { gchar *name; gchar *file_name; GtkWidget *fs; gint response; AgWindowPrivate *priv = ag_window_get_instance_private(window); ag_window_recalculate_chart(window, FALSE); // We should never enter here, but who knows... if (priv->chart == NULL) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, _("Chart cannot be calculated.") ); g_set_error( err, AG_WINDOW_ERROR, AG_WINDOW_ERROR_EMPTY_CHART, "Chart is empty" ); return; } name = ag_chart_get_name(priv->chart); if ((name == NULL) || (*name == 0)) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, _("You must enter a name before saving a chart.") ); g_set_error( err, AG_WINDOW_ERROR, AG_WINDOW_ERROR_NO_NAME, "No name specified" ); return; } file_name = g_strdup_printf("%s.agc", name); fs = gtk_file_chooser_dialog_new(_("Export Chart"), GTK_WINDOW(window), GTK_FILE_CHOOSER_ACTION_SAVE, _("_Cancel"), GTK_RESPONSE_CANCEL, _("_Save"), GTK_RESPONSE_ACCEPT, NULL); gtk_dialog_set_default_response(GTK_DIALOG(fs), GTK_RESPONSE_ACCEPT); gtk_file_chooser_set_local_only(GTK_FILE_CHOOSER(fs), FALSE); gtk_file_chooser_set_do_overwrite_confirmation(GTK_FILE_CHOOSER(fs), TRUE); gtk_file_chooser_set_current_name(GTK_FILE_CHOOSER(fs), file_name); g_free(file_name); response = gtk_dialog_run(GTK_DIALOG(fs)); gtk_widget_hide(fs); if (response == GTK_RESPONSE_ACCEPT) { GFile *file = gtk_file_chooser_get_file(GTK_FILE_CHOOSER(fs)); ag_chart_save_to_file(priv->chart, file, err); } gtk_widget_destroy(fs); } static void ag_window_export_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { AgWindow *window = AG_WINDOW(user_data); GError *err = NULL; ag_window_recalculate_chart(window, TRUE); ag_window_export(window, &err); if (err) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, "%s", err->message ); } } gboolean ag_window_can_close(AgWindow *window, gboolean display_dialog) { AgWindowPrivate *priv = ag_window_get_instance_private(window); gint db_id = (priv->saved_data) ? priv->saved_data->db_id : -1; AgDbSave *save_data = NULL; AgDb *db = ag_db_get(); GError *err = NULL; gboolean ret = TRUE; if (priv->chart) { ag_window_recalculate_chart(window, TRUE); save_data = ag_chart_get_db_save(priv->chart, db_id); if ( !ag_db_save_identical(priv->saved_data, save_data, FALSE) || !(priv->saved_data) || (priv->saved_data->db_id == -1) ) { g_debug("Save is needed!"); if (display_dialog) { gint response; response = ag_app_buttoned_dialog( GTK_WIDGET(window), GTK_MESSAGE_QUESTION, _("Chart is not saved. Do you want to save it?"), _("Save and close"), GTK_RESPONSE_YES, _("Close without saving"), GTK_RESPONSE_NO, _("Return to chart"), GTK_RESPONSE_CANCEL, NULL ); switch (response) { case GTK_RESPONSE_YES: if (!ag_db_save_chart(db, save_data, &err)) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, "Unable to save chart: %s", err->message ); ret = FALSE; } else { ret = TRUE; } break; case GTK_RESPONSE_NO: ret = TRUE; break; default: ret = FALSE; break; } } else { ret = FALSE; } } } ag_db_save_data_free(save_data); return ret; } static void ag_window_save_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { AgWindow *window = AG_WINDOW(user_data); AgWindowPrivate *priv = ag_window_get_instance_private(window); AgDb *db = ag_db_get(); GError *err = NULL; gint old_id; AgDbSave *save_data; ag_window_recalculate_chart(window, TRUE); if (!ag_window_can_close(window, FALSE)) { old_id = (priv->saved_data) ? priv->saved_data->db_id : -1; save_data = ag_chart_get_db_save(priv->chart, old_id); if (!ag_db_save_chart(db, save_data, &err)) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, _("Unable to save: %s"), err->message ); } ag_db_save_data_free(priv->saved_data); priv->saved_data = save_data; } } static void ag_window_close_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { AgWindow *window = AG_WINDOW(user_data); if (ag_window_can_close(window, TRUE)) { gtk_widget_destroy(GTK_WIDGET(window)); } } static gboolean ag_window_delete_event_callback(AgWindow *window, GdkEvent *event, gpointer user_data) { return (!ag_window_can_close(window, TRUE)); } static void ag_window_tab_changed_cb(GtkStack *stack, GParamSpec *pspec, AgWindow *window) { GtkWidget *active_tab; const gchar *active_tab_name = gtk_stack_get_visible_child_name(stack); AgWindowPrivate *priv = ag_window_get_instance_private(window); g_debug("Active tab changed: %s", active_tab_name); if (active_tab_name == NULL) { return; } active_tab = gtk_stack_get_visible_child(stack); if (strcmp("chart", active_tab_name) == 0) { gtk_widget_set_size_request(active_tab, 600, 600); } if (strcmp("list", active_tab_name) == 0) { gtk_stack_set_visible_child_name( GTK_STACK(priv->menubutton_stack), "list" ); gtk_stack_set_visible_child_name( GTK_STACK(priv->new_back_stack), "new" ); } else { gtk_stack_set_visible_child_name( GTK_STACK(priv->menubutton_stack), "chart" ); gtk_stack_set_visible_child_name( GTK_STACK(priv->new_back_stack), "back" ); // Note that priv->current_tab is actually the previously selected tab, // not the real active one! if (priv->current_tab == priv->tab_edit) { ag_window_recalculate_chart(window, FALSE); ag_window_redraw_chart(window); } } priv->current_tab = active_tab; } static void ag_window_change_tab_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { AgWindow *window = AG_WINDOW(user_data); const gchar *target_tab = g_variant_get_string(parameter, NULL); AgWindowPrivate *priv = ag_window_get_instance_private(window); gtk_stack_set_visible_child_name(GTK_STACK(priv->stack), target_tab); g_action_change_state(G_ACTION(action), parameter); } static gboolean ag_window_set_default_house_system(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, AgWindow *window) { GsweHouseSystem row_house_system; AgWindowPrivate *priv = ag_window_get_instance_private(window); AgSettings *settings = ag_settings_get(); GSettings *main_settings = ag_settings_peek_main_settings(settings); GsweHouseSystem house_system = g_settings_get_enum( main_settings, "default-house-system" ); g_object_unref(settings); gtk_tree_model_get( GTK_TREE_MODEL(priv->house_system_model), iter, 0, &row_house_system, -1 ); if (house_system == row_house_system) { gtk_combo_box_set_active_iter(GTK_COMBO_BOX(priv->house_system), iter); return TRUE; } return FALSE; } static void ag_window_new_chart_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { AgWindow *window = AG_WINDOW(user_data); AgWindowPrivate *priv = ag_window_get_instance_private(window); /* Empty edit tab values */ gtk_entry_set_text(GTK_ENTRY(priv->name), ""); gtk_entry_set_text(GTK_ENTRY(priv->country), ""); gtk_entry_set_text(GTK_ENTRY(priv->city), ""); gtk_spin_button_set_value(GTK_SPIN_BUTTON(priv->year), (gdouble)1); gtk_spin_button_set_value(GTK_SPIN_BUTTON(priv->month), (gdouble)1); gtk_spin_button_set_value(GTK_SPIN_BUTTON(priv->day), (gdouble)1); gtk_spin_button_set_value(GTK_SPIN_BUTTON(priv->hour), (gdouble)1); gtk_spin_button_set_value(GTK_SPIN_BUTTON(priv->minute), (gdouble)1); gtk_spin_button_set_value(GTK_SPIN_BUTTON(priv->second), (gdouble)1); gtk_spin_button_set_value(GTK_SPIN_BUTTON(priv->timezone), 0.0); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(priv->north_lat), TRUE); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(priv->east_long), TRUE); gtk_spin_button_set_value(GTK_SPIN_BUTTON(priv->longitude), 0.0); gtk_spin_button_set_value(GTK_SPIN_BUTTON(priv->latitude), 0.0); gtk_tree_model_foreach( GTK_TREE_MODEL(priv->house_system_model), (GtkTreeModelForeachFunc)ag_window_set_default_house_system, window ); gtk_text_buffer_set_text(GTK_TEXT_BUFFER(priv->note_buffer), "", 0); if (priv->chart) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, "This window already has a chart. " \ "This should not happen, " \ "please consider issuing a bug report!" ); gtk_stack_set_visible_child_name(GTK_STACK(priv->stack), "chart"); return; } gtk_stack_set_visible_child_name(GTK_STACK(priv->stack), "edit"); } static void ag_window_back_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { AgWindow *window = AG_WINDOW(user_data); AgWindowPrivate *priv = ag_window_get_instance_private(window); g_debug("Back button pressed"); if (ag_window_can_close(window, TRUE)) { g_clear_object(&(priv->chart)); ag_db_save_data_free(priv->saved_data); priv->saved_data = NULL; ag_window_load_chart_list(window); gtk_stack_set_visible_child_name(GTK_STACK(priv->stack), "list"); gtk_header_bar_set_subtitle(GTK_HEADER_BAR(priv->header_bar), NULL); } } static void ag_window_refresh_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { ag_window_load_chart_list(AG_WINDOW(user_data)); } static void ag_window_selection_mode_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { GVariant *state; gboolean new_state; GtkStyleContext *style; AgWindow *window = AG_WINDOW(user_data); AgWindowPrivate *priv = ag_window_get_instance_private(window); state = g_action_get_state(G_ACTION(action)); new_state = !g_variant_get_boolean(state); g_action_change_state(G_ACTION(action), g_variant_new_boolean(new_state)); g_variant_unref(state); style = gtk_widget_get_style_context(priv->header_bar); if (new_state) { gtk_header_bar_set_show_close_button( GTK_HEADER_BAR(priv->header_bar), FALSE ); gtk_style_context_add_class(style, "selection-mode"); gd_main_view_set_selection_mode(GD_MAIN_VIEW(priv->tab_list), TRUE); gtk_widget_hide(priv->new_back_stack); gtk_stack_set_visible_child_name( GTK_STACK(priv->menubutton_stack), "selection" ); } else { gtk_header_bar_set_show_close_button( GTK_HEADER_BAR(priv->header_bar), TRUE ); gtk_style_context_remove_class(style, "selection-mode"); gd_main_view_set_selection_mode(GD_MAIN_VIEW(priv->tab_list), FALSE); gtk_widget_show_all(priv->new_back_stack); gtk_stack_set_visible_child_name( GTK_STACK(priv->menubutton_stack), "list" ); } } static void ag_window_delete_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { GList *selection, *item; GtkTreeModel *model; AgWindow *window = AG_WINDOW(user_data); AgWindowPrivate *priv = ag_window_get_instance_private(window); AgDb *db = ag_db_get(); selection = gd_main_view_get_selection(GD_MAIN_VIEW(priv->tab_list)); model = gd_main_view_get_model(GD_MAIN_VIEW(priv->tab_list)); for (item = selection; item; item = g_list_next(item)) { GtkTreePath *path = item->data; GtkTreeIter iter; gchar *id_str; gint id; GError *err = NULL; gtk_tree_model_get_iter(model, &iter, path); gtk_tree_model_get( model, &iter, GD_MAIN_COLUMN_ID, &id_str, -1 ); id = atoi(id_str); g_free(id_str); if (!ag_db_delete_chart(db, id, &err)) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, "Unable to delete chart: %s", (err && err->message) ? err->message : "No reason" ); } } g_action_group_activate_action(G_ACTION_GROUP(window), "selection", NULL); g_action_group_activate_action(G_ACTION_GROUP(window), "refresh", NULL); } static void ag_window_js_callback(GObject *object, GAsyncResult *res, gpointer user_data) { WebKitJavascriptResult *js_result; GError *err = NULL; if ((js_result = webkit_web_view_run_javascript_finish( WEBKIT_WEB_VIEW(object), res, &err )) != NULL) { webkit_javascript_result_unref(js_result); } } static void ag_window_connection_action(GSimpleAction *action, GVariant *parameter, gpointer user_data) { GVariant *current_state; const gchar *state; gchar *js_code = NULL; AgWindowPrivate *priv = ag_window_get_instance_private( AG_WINDOW(user_data) ); static gchar *js = "aspects = document.getElementById('aspects');\n" \ "antiscia = document.getElementById('antiscia');\n" \ "aspects.setAttribute('visibility', '%s');\n" \ "antiscia.setAttribute('visibility', '%s');\n"; current_state = g_action_get_state(G_ACTION(action)); if (g_variant_equal(current_state, parameter)) { return; } g_action_change_state(G_ACTION(action), parameter); state = g_variant_get_string(parameter, NULL); if (strcmp("aspects", state) == 0) { g_debug("Switching to aspects"); js_code = g_strdup_printf(js, "visible", "hidden"); } else if (strcmp("antiscia", state) == 0) { g_debug("Switching to antiscia"); js_code = g_strdup_printf(js, "hidden", "visible"); } else { g_warning("Connection type '%s' is invalid", state); } if (js_code) { webkit_web_view_run_javascript( WEBKIT_WEB_VIEW(priv->chart_web_view), js_code, NULL, ag_window_js_callback, NULL ); g_free(js_code); } } static GActionEntry win_entries[] = { { "close", ag_window_close_action, NULL, NULL, NULL }, { "save", ag_window_save_action, NULL, NULL, NULL }, { "export", ag_window_export_action, NULL, NULL, NULL }, { "export-svg", ag_window_export_svg_action, NULL, NULL, NULL }, { "view-menu", ag_window_view_menu_action, NULL, "false", NULL }, { "gear-menu", ag_window_gear_menu_action, NULL, "false", NULL }, { "change-tab", ag_window_change_tab_action, "s", "'edit'", NULL }, { "new-chart", ag_window_new_chart_action, NULL, NULL, NULL }, { "back", ag_window_back_action, NULL, NULL, NULL }, { "refresh", ag_window_refresh_action, NULL, NULL, NULL }, { "selection", ag_window_selection_mode_action, NULL, "false", NULL }, { "delete", ag_window_delete_action, NULL, NULL, NULL }, { "connection", ag_window_connection_action, "s", "'aspects'", NULL }, }; static void ag_window_display_changed(GSettings *settings, gchar *key, AgWindow *window) { AgWindowPrivate *priv = ag_window_get_instance_private(window); /* The planet symbols are redrawn only if aspect_table_populated is * set to FALSE */ if (g_str_equal("planets-char", key)) { priv->aspect_table_populated = FALSE; } ag_window_redraw_aspect_table(window); } static void ag_window_add_house_system(GsweHouseSystemInfo *house_system_info, AgWindowPrivate *priv) { GtkTreeIter iter; gtk_list_store_append(priv->house_system_model, &iter); gtk_list_store_set( priv->house_system_model, &iter, 0, gswe_house_system_info_get_house_system(house_system_info), 1, gswe_house_system_info_get_name(house_system_info), -1 ); } static void ag_window_list_item_activated_cb(GdMainView *view, const gchar *id, const GtkTreePath *path, AgWindow *window) { guint row_id = atoi(id); AgWindowPrivate *priv = ag_window_get_instance_private(window); AgDb *db = ag_db_get(); GError *err = NULL; if (priv->saved_data != NULL) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, "Window chart is not saved. " \ "This is a bug, it should not happen here. " \ "Please consider opening a bug report!" ); ag_window_change_tab(window, "chart"); return; } g_debug("Loading chart with ID %d", row_id); if ((priv->saved_data = ag_db_get_chart_data_by_id( db, row_id, &err)) == NULL) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, "Could not open chart." ); return; } if (priv->chart) { g_clear_object(&(priv->chart)); } if ((priv->chart = ag_chart_new_from_db_save( priv->saved_data, &err )) == NULL) { ag_app_message_dialog( GTK_WIDGET(window), GTK_MESSAGE_ERROR, "Error: %s", err->message ); ag_db_save_data_free(priv->saved_data); priv->saved_data = NULL; return; } ag_window_update_from_chart(window); ag_window_change_tab(window, "chart"); } static void ag_window_list_selection_changed_cb(GdMainView *view, AgWindow *window) { GList *selection; guint count; AgWindowPrivate *priv = ag_window_get_instance_private(window); selection = gd_main_view_get_selection(view); if ((count = g_list_length(selection)) > 0) { gtk_revealer_set_reveal_child( GTK_REVEALER(priv->selection_toolbar), TRUE ); } else { gtk_revealer_set_reveal_child( GTK_REVEALER(priv->selection_toolbar), FALSE ); } // Here it is possible to set button sensitivity later } static gboolean ag_window_city_matches(GtkEntryCompletion *city_comp, const gchar *key, GtkTreeIter *iter, AgWindow *window) { AgWindowPrivate *priv = ag_window_get_instance_private(window); gchar *ccode, *name, *normalized_name, *case_normalized_name; gboolean ret = FALSE; gtk_tree_model_get( gtk_entry_completion_get_model(city_comp), iter, AG_CITY_NAME, &name, AG_CITY_COUNTRY, &ccode, -1 ); if ( (priv->selected_country == NULL) || (strcmp(priv->selected_country, ccode) == 0) ) { normalized_name = g_utf8_normalize(name, -1, G_NORMALIZE_ALL); if (normalized_name) { case_normalized_name = g_utf8_casefold(normalized_name, -1); if (strncmp(key, case_normalized_name, strlen(key)) == 0) { ret = TRUE; } g_free(case_normalized_name); g_free(normalized_name); } } g_free(name); g_free(ccode); return ret; } static void ag_window_init(AgWindow *window) { GtkAccelGroup *accel_group; GSettings *main_settings; GList *house_system_list; GtkCellRenderer *house_system_renderer; AgWindowPrivate *priv = ag_window_get_instance_private(window); gtk_widget_init_template(GTK_WIDGET(window)); priv->settings = ag_settings_get(); main_settings = ag_settings_peek_main_settings(priv->settings); g_signal_connect( G_OBJECT(main_settings), "changed::planets-char", G_CALLBACK(ag_window_display_changed), window ); g_signal_connect( G_OBJECT(main_settings), "changed::aspects-char", G_CALLBACK(ag_window_display_changed), window ); gtk_entry_completion_set_model(priv->country_comp, country_list); gtk_entry_completion_set_text_column(priv->country_comp, AG_COUNTRY_NAME); gtk_entry_set_completion(GTK_ENTRY(priv->country), priv->country_comp); gtk_entry_completion_set_model(priv->city_comp, city_list); gtk_entry_completion_set_text_column(priv->city_comp, AG_CITY_NAME); gtk_entry_completion_set_minimum_key_length(priv->city_comp, 3); gtk_entry_set_completion(GTK_ENTRY(priv->city), priv->city_comp); gtk_entry_completion_set_match_func( priv->city_comp, (GtkEntryCompletionMatchFunc)ag_window_city_matches, window, NULL ); house_system_list = gswe_all_house_systems(); g_list_foreach(house_system_list, (GFunc)ag_window_add_house_system, priv); g_list_free(house_system_list); gtk_tree_model_foreach( GTK_TREE_MODEL(priv->house_system_model), (GtkTreeModelForeachFunc)ag_window_set_default_house_system, window ); house_system_renderer = gtk_cell_renderer_text_new(); gtk_cell_layout_pack_start( GTK_CELL_LAYOUT(priv->house_system), house_system_renderer, TRUE ); gtk_cell_layout_set_attributes( GTK_CELL_LAYOUT(priv->house_system), house_system_renderer, "text", 1, NULL ); priv->tab_list = GTK_WIDGET(gd_main_view_new(GD_MAIN_VIEW_ICON)); gtk_stack_add_titled( GTK_STACK(priv->stack), priv->tab_list, "list", "Chart list" ); gd_main_view_set_selection_mode(GD_MAIN_VIEW(priv->tab_list), FALSE); gd_main_view_set_model( GD_MAIN_VIEW(priv->tab_list), GTK_TREE_MODEL(priv->db_chart_data) ); g_signal_connect( priv->tab_list, "item-activated", G_CALLBACK(ag_window_list_item_activated_cb), window ); g_signal_connect( priv->tab_list, "view-selection-changed", G_CALLBACK(ag_window_list_selection_changed_cb), window ); gtk_stack_set_visible_child_name(GTK_STACK(priv->stack), "list"); priv->current_tab = priv->tab_list; g_object_set( priv->year_adjust, "lower", (gdouble)G_MININT, "upper", (gdouble)G_MAXINT, NULL ); priv->chart = NULL; g_action_map_add_action_entries( G_ACTION_MAP(window), win_entries, G_N_ELEMENTS(win_entries), window ); accel_group = gtk_accel_group_new(); gtk_window_add_accel_group(GTK_WINDOW(window), accel_group); } static void ag_window_dispose(GObject *gobject) { AgWindowPrivate *priv = ag_window_get_instance_private(AG_WINDOW(gobject)); g_clear_object(&priv->settings); G_OBJECT_CLASS(ag_window_parent_class)->dispose(gobject); } static void ag_window_name_changed_cb(GtkEntry *name_entry, AgWindow *window) { const gchar *name; AgWindowPrivate *priv = ag_window_get_instance_private(window); name = gtk_entry_get_text(name_entry); gtk_header_bar_set_subtitle(GTK_HEADER_BAR(priv->header_bar), name); } static gboolean ag_window_find_country(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, struct cc_search *search) { gchar *name, *ccode; gboolean found = FALSE; gtk_tree_model_get( model, iter, AG_COUNTRY_NAME, &name, AG_COUNTRY_CODE, &ccode, -1 ); if (g_utf8_collate(search->target, name) == 0) { found = TRUE; search->ret_iter = gtk_tree_iter_copy(iter); search->ret_code = ccode; } else { g_free(ccode); } return found; } /** * ag_window_country_changed_callback: * @country: the #GtkSearchEntry for country search * @window: the window in which the event happens * * This function is called whenever the text in the country search entry is * changed. */ static void ag_window_country_changed_callback(GtkSearchEntry *country, AgWindow *window) { struct cc_search search; AgWindowPrivate *priv = ag_window_get_instance_private(window); search.target = gtk_entry_get_text(GTK_ENTRY(country)); search.ret_iter = NULL; gtk_tree_model_foreach( country_list, (GtkTreeModelForeachFunc)ag_window_find_country, &search ); g_free(priv->selected_country); if (search.ret_iter != NULL) { g_debug("Country (entry-changed): %s", search.ret_code); gtk_tree_iter_free(search.ret_iter); priv->selected_country = search.ret_code; } else { priv->selected_country = NULL; } } static gboolean ag_window_find_city(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, struct cc_search *search) { gchar *name, *ccode; gboolean found = FALSE; gtk_tree_model_get( model, iter, AG_CITY_NAME, &name, AG_CITY_COUNTRY, &ccode, -1 ); if (g_utf8_collate(search->target, name) == 0) { found = TRUE; search->ret_iter = gtk_tree_iter_copy(iter); search->ret_code = ccode; } else { g_free(ccode); } return found; } static void ag_window_city_changed_callback(GtkSearchEntry *city, AgWindow *window) { struct cc_search search; AgWindowPrivate *priv = ag_window_get_instance_private(window); search.target = gtk_entry_get_text(GTK_ENTRY(city)); search.ret_iter = NULL; gtk_tree_model_foreach( city_list, (GtkTreeModelForeachFunc)ag_window_find_city, &search ); g_free(priv->selected_city); if (search.ret_iter != NULL) { gdouble longitude, latitude, altitude; gchar *name, *ccode; gtk_tree_model_get( city_list, search.ret_iter, AG_CITY_COUNTRY, &ccode, AG_CITY_NAME, &name, AG_CITY_LAT, &latitude, AG_CITY_LONG, &longitude, AG_CITY_ALT, &altitude, -1 ); if ( (priv->selected_country != NULL) && (strcmp(priv->selected_country, ccode) != 0) ) { return; } if (latitude < 0.0) { gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(priv->south_lat), TRUE ); gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->latitude), -latitude ); } else { gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(priv->north_lat), TRUE ); gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->latitude), latitude ); } if (longitude < 0.0) { gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(priv->west_long), TRUE ); gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->longitude), -longitude ); } else { gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(priv->east_long), TRUE ); gtk_spin_button_set_value( GTK_SPIN_BUTTON(priv->longitude), longitude ); } // TODO: implement setting altitude maybe? Is that really necessary? g_debug("City (entry-changed): %s (%s); %.6f, %.6f, %.6f", name, search.ret_code, longitude, latitude, altitude); g_free(name); gtk_tree_iter_free(search.ret_iter); priv->selected_city = search.ret_code; } else { priv->selected_city = NULL; } } static void ag_window_class_init(AgWindowClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS(klass); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS(klass); gobject_class->dispose = ag_window_dispose; gtk_widget_class_set_template_from_resource( widget_class, "/eu/polonkai/gergely/Astrognome/ui/ag-window.ui" ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, header_bar ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, new_back_stack ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, menubutton_stack ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, db_chart_data ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, tab_edit ); gtk_widget_class_bind_template_child_private(widget_class, AgWindow, name); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, country ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, country_comp ); gtk_widget_class_bind_template_child_private(widget_class, AgWindow, city); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, city_comp ); gtk_widget_class_bind_template_child_private(widget_class, AgWindow, year); gtk_widget_class_bind_template_child_private(widget_class, AgWindow, month); gtk_widget_class_bind_template_child_private(widget_class, AgWindow, day); gtk_widget_class_bind_template_child_private(widget_class, AgWindow, hour); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, minute ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, second ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, timezone ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, north_lat ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, south_lat ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, east_long ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, west_long ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, latitude ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, longitude ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, house_system_model ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, house_system ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, tab_chart ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, aspect_table ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, year_adjust ); gtk_widget_class_bind_template_child_private(widget_class, AgWindow, stack); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, note_buffer ); gtk_widget_class_bind_template_child_private( widget_class, AgWindow, selection_toolbar ); gtk_widget_class_bind_template_callback( widget_class, ag_window_delete_event_callback ); gtk_widget_class_bind_template_callback( widget_class, ag_window_tab_changed_cb ); gtk_widget_class_bind_template_callback( widget_class, ag_window_name_changed_cb ); gtk_widget_class_bind_template_callback( widget_class, ag_window_country_changed_callback ); gtk_widget_class_bind_template_callback( widget_class, ag_window_city_changed_callback ); } gboolean ag_window_chart_context_cb(WebKitWebView *web_view, GtkWidget *default_menu, WebKitHitTestResult *hit_test_result, gboolean triggered_with_keyboard, gpointer user_data) { return TRUE; } static gboolean ag_window_configure_event_cb(GtkWidget *widget, GdkEventConfigure *event, gpointer user_data) { AgWindow *window = AG_WINDOW(widget); AgWindowPrivate *priv = ag_window_get_instance_private(window); ag_window_settings_save( GTK_WINDOW(window), ag_settings_peek_window_settings(priv->settings) ); return FALSE; } GtkWidget * ag_window_new(AgApp *app, WebKitUserContentManager *manager) { AgWindow *window = g_object_new(AG_TYPE_WINDOW, NULL); AgWindowPrivate *priv = ag_window_get_instance_private(window); priv->chart_web_view = webkit_web_view_new_with_user_content_manager( manager ); gtk_box_pack_end( GTK_BOX(priv->tab_chart), priv->chart_web_view, TRUE, TRUE, 0 ); g_signal_connect( priv->chart_web_view, "context-menu", G_CALLBACK(ag_window_chart_context_cb), NULL ); // TODO: translate this error message! webkit_web_view_load_html( WEBKIT_WEB_VIEW(priv->chart_web_view), "" \ "" \ "No Chart" \ "" \ "" \ "

No Chart

" \ "

No chart is loaded. Create one on the " \ "edit view, or open one from the application menu!

" \ "" \ "", NULL); gtk_window_set_application(GTK_WINDOW(window), GTK_APPLICATION(app)); gtk_window_set_icon_name(GTK_WINDOW(window), "astrognome"); g_signal_connect( window, "configure-event", G_CALLBACK(ag_window_configure_event_cb), NULL ); ag_window_settings_restore( GTK_WINDOW(window), ag_settings_peek_window_settings(priv->settings) ); return GTK_WIDGET(window); } void ag_window_set_chart(AgWindow *window, AgChart *chart) { AgWindowPrivate *priv = ag_window_get_instance_private(window); if (priv->chart != NULL) { g_signal_handlers_disconnect_by_func( priv->chart, ag_window_chart_changed, window ); g_clear_object(&(priv->chart)); } ag_db_save_data_free(priv->saved_data); priv->chart = chart; g_signal_connect( priv->chart, "changed", G_CALLBACK(ag_window_chart_changed), window ); g_object_ref(chart); priv->saved_data = ag_chart_get_db_save(chart, -1); } AgChart * ag_window_get_chart(AgWindow *window) { AgWindowPrivate *priv = ag_window_get_instance_private(window); return priv->chart; } void ag_window_settings_restore(GtkWindow *window, GSettings *settings) { gint width, height; gboolean maximized; GdkScreen *screen; width = g_settings_get_int(settings, "width"); height = g_settings_get_int(settings, "height"); maximized = g_settings_get_boolean(settings, "maximized"); if ((width > 1) && (height > 1)) { gint max_width, max_height; screen = gtk_widget_get_screen(GTK_WIDGET(window)); max_width = gdk_screen_get_width(screen); max_height = gdk_screen_get_height(screen); width = CLAMP(width, 0, max_width); height = CLAMP(height, 0, max_height); gtk_window_set_default_size(window, width, height); } if (maximized) { gtk_window_maximize(window); } } void ag_window_settings_save(GtkWindow *window, GSettings *settings) { GdkWindowState state; gint width, height; gboolean maximized; state = gdk_window_get_state(gtk_widget_get_window(GTK_WIDGET(window))); maximized = ( (state & GDK_WINDOW_STATE_MAXIMIZED) == GDK_WINDOW_STATE_MAXIMIZED ); g_settings_set_boolean(settings, "maximized", maximized); gtk_window_get_size(window, &width, &height); g_settings_set_int(settings, "width", width); g_settings_set_int(settings, "height", height); } void ag_window_change_tab(AgWindow *window, const gchar *tab_name) { AgWindowPrivate *priv = ag_window_get_instance_private(window); gtk_stack_set_visible_child_name(GTK_STACK(priv->stack), tab_name); g_action_change_state( g_action_map_lookup_action(G_ACTION_MAP(window), "change-tab"), g_variant_new_string(tab_name) ); } static void ag_window_add_chart_to_list(AgDbSave *save_data, AgWindow *window) { GtkTreeIter iter; AgWindowPrivate *priv = ag_window_get_instance_private(window); gchar *id = g_strdup_printf("%d", save_data->db_id); gtk_list_store_append(priv->db_chart_data, &iter); gtk_list_store_set( priv->db_chart_data, &iter, 0, id, /* ID */ 1, NULL, /* URI */ 2, save_data->name, /* Primary text */ 3, NULL, /* Secondary text */ 4, NULL, /* Icon */ 5, 0, /* mtime */ 6, FALSE, /* Selected */ 7, 0, /* Pulse */ -1 ); g_free(id); } static void ag_window_clear_chart_list(AgWindow *window) { AgWindowPrivate *priv = ag_window_get_instance_private(window); gtk_list_store_clear(priv->db_chart_data); } gboolean ag_window_load_chart_list(AgWindow *window) { AgDb *db = ag_db_get(); GError *err = NULL; GList *chart_list = ag_db_get_chart_list(db, &err); ag_window_clear_chart_list(window); /* With only a few charts, this should be fine. Maybe implementing lazy * loading would be a better idea. See: * http://blogs.gnome.org/ebassi/documentation/lazy-loading/ */ g_list_foreach(chart_list, (GFunc)ag_window_add_chart_to_list, window); return TRUE; }