#include #include #include #include #include #include #include #include #include "mmyth_ui.h" #include "mmyth_uicommon.h" #include "mmyth_recordui.h" #include "mmyth_schedulerui.h" /* * GMyth library includes */ #include #include #include static void run_calendar_dialog(GtkButton * button, gpointer data); static void add_channel_field(MMythSchedulerUI * scheduler_ui, GtkWidget * vbox); static void add_time_field(MMythSchedulerUI * scheduler_ui, GtkWidget * vbox); static void add_date_field(MMythSchedulerUI * scheduler_ui, GtkWidget * vbox); static void add_duration_field(MMythSchedulerUI * scheduler_ui, GtkWidget * vbox); static void add_frequency_field(MMythSchedulerUI * scheduler_ui, GtkWidget * vbox); static void add_title_field(MMythSchedulerUI * scheduler_ui, GtkWidget * vbox); MMythSchedulerUI * mmyth_schedulerui_new(GMythBackendInfo * backend_info) { GtkWidget *scrolledwindow; GtkWidget *viewport; GtkWidget *head_hbox; GtkWidget *fields_vbox; GtkWidget *hseparator; GtkWidget *label; MMythSchedulerUI *scheduler_ui = g_new0(MMythSchedulerUI, 1); scheduler_ui->backend_info = backend_info; scrolledwindow = gtk_scrolled_window_new(NULL, NULL); scheduler_ui->main_widget = scrolledwindow; gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolledwindow), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); // Is this needed? viewport = gtk_viewport_new(NULL, NULL); gtk_container_add(GTK_CONTAINER(scrolledwindow), viewport); // Is this needed? head_hbox = gtk_hbox_new(FALSE, 0); gtk_container_add(GTK_CONTAINER(viewport), head_hbox); fields_vbox = gtk_vbox_new(FALSE, 0); gtk_box_pack_start(GTK_BOX(head_hbox), fields_vbox, TRUE, TRUE, 0); gtk_container_set_border_width(GTK_CONTAINER(fields_vbox), 10); label = gtk_label_new_with_mnemonic(("Manual Schedule Recording")); gtk_box_pack_start(GTK_BOX(fields_vbox), label, FALSE, FALSE, 0); gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); hseparator = gtk_hseparator_new(); gtk_box_pack_start(GTK_BOX(fields_vbox), hseparator, FALSE, TRUE, 0); add_channel_field(scheduler_ui, fields_vbox); add_time_field(scheduler_ui, fields_vbox); add_date_field(scheduler_ui, fields_vbox); add_duration_field(scheduler_ui, fields_vbox); add_frequency_field(scheduler_ui, fields_vbox); add_title_field(scheduler_ui, fields_vbox); return scheduler_ui; } static void set_date_from_calendar(GtkCalendar * calendar, gpointer data) { char sched_date[24]; MMythSchedulerUI *scheduler_ui = (MMythSchedulerUI *) data; // FIXME: Change this, save another value instead of month_temp, // day_temp, ... gtk_calendar_get_date(GTK_CALENDAR(calendar), &(scheduler_ui->year_temp), &(scheduler_ui->month_temp), &(scheduler_ui->day_temp)); sched_date[23] = '\0'; g_sprintf(sched_date, "%04d %02d %02d (yyyy mm dd)", scheduler_ui->year_temp, scheduler_ui->month_temp + 1, scheduler_ui->day_temp); gtk_button_set_label(GTK_BUTTON(scheduler_ui->date_button), sched_date); gtk_widget_destroy(scheduler_ui->calendar_dialog); scheduler_ui->calendar_dialog = NULL; scheduler_ui->calendar = NULL; } // calendar static void run_calendar_dialog(GtkButton * button, gpointer data) { GtkWidget *dialog_vbox; MMythSchedulerUI *scheduler_ui = (MMythSchedulerUI *) data; // calendar_dialog and calendar are been released at // set_date_from_calendar () scheduler_ui->calendar_dialog = gtk_dialog_new(); gtk_container_set_border_width(GTK_CONTAINER (scheduler_ui->calendar_dialog), 1); gtk_window_set_title(GTK_WINDOW(scheduler_ui->calendar_dialog), "Select starting date"); gtk_window_set_position(GTK_WINDOW(scheduler_ui->calendar_dialog), GTK_WIN_POS_CENTER); gtk_window_set_decorated(GTK_WINDOW(scheduler_ui->calendar_dialog), FALSE); dialog_vbox = GTK_DIALOG(scheduler_ui->calendar_dialog)->vbox; scheduler_ui->calendar = gtk_calendar_new(); gtk_box_pack_start(GTK_BOX(dialog_vbox), scheduler_ui->calendar, TRUE, TRUE, 0); gtk_calendar_display_options(GTK_CALENDAR(scheduler_ui->calendar), GTK_CALENDAR_SHOW_HEADING | GTK_CALENDAR_SHOW_DAY_NAMES); gtk_widget_show_all(scheduler_ui->calendar_dialog); g_signal_connect(G_OBJECT(scheduler_ui->calendar), "day-selected-double-click", G_CALLBACK(set_date_from_calendar), data); } gboolean mmyth_schedulerui_save(MMythSchedulerUI * scheduler_ui) { GMythScheduler *scheduler; ScheduleInfo *schedule_info; GMythChannelInfo *channel_info; GList *clist; gint index, duration; gint frequency; struct tm start_tm; schedule_info = g_new0(ScheduleInfo, 1); if (schedule_info == NULL) { g_warning("Error allocating memory"); return FALSE; } clist = scheduler_ui->channel_list; index = gtk_combo_box_get_active(GTK_COMBO_BOX (scheduler_ui->channel_combobox)); if (clist != NULL) clist = g_list_nth(clist, index); if (clist) { g_debug("[%s] New schedule: %d", __FUNCTION__, index); } else { g_warning("[%s] Error when adding new schedule", __FUNCTION__); return FALSE; } channel_info = clist->data; /* * initialize schedule_info */ schedule_info->channel_id = channel_info->channel_ID; start_tm.tm_hour = (int) gtk_spin_button_get_value(GTK_SPIN_BUTTON (scheduler_ui->hour_spinbutton)); start_tm.tm_min = (int) gtk_spin_button_get_value(GTK_SPIN_BUTTON (scheduler_ui->min_spinbutton)); start_tm.tm_sec = 0; start_tm.tm_mday = (gint) scheduler_ui->day_temp; start_tm.tm_mon = (gint) scheduler_ui->month_temp; start_tm.tm_year = (gint) scheduler_ui->year_temp - 1900; // years // since // 1900 GTimeVal *time_val_local = g_new0(GTimeVal, 1); time_val_local->tv_sec = timelocal(&start_tm); schedule_info->start_time = time_val_local; if (NULL == schedule_info->start_time) { g_warning("timelocal error!\n"); return FALSE; } duration = (gint) gtk_spin_button_get_value(GTK_SPIN_BUTTON (scheduler_ui->duration_spinbutton)); schedule_info->end_time = schedule_info->start_time + (duration * 60); /* * TODO: frequency is not implemented yet */ frequency = gtk_combo_box_get_active(GTK_COMBO_BOX (scheduler_ui->freq_combobox)); schedule_info->title = g_string_new(""); g_string_printf(schedule_info->title, "%s", gtk_entry_get_text(GTK_ENTRY (scheduler_ui->title_entry))); /* * FIXME: Architecture change to reuse the scheduler created in the * recordui! */ scheduler = gmyth_scheduler_new(); gmyth_scheduler_connect(scheduler, scheduler->backend_info); /* * FIXME: set record_id = -1 to add a new schedule */ schedule_info->record_id = -1; gmyth_scheduler_add_schedule(scheduler, schedule_info); gmyth_scheduler_disconnect(scheduler); /* * free allocated memory */ g_object_unref(scheduler); g_free(schedule_info); return TRUE; } static GtkWidget * add_line(GtkWidget * vbox, const gchar * str) { GtkWidget *label; GtkWidget *hbox = gtk_hbox_new(FALSE, 0); gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0); gtk_container_set_border_width(GTK_CONTAINER(hbox), 3); label = gtk_label_new(str); gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); return hbox; } static void add_channel_field(MMythSchedulerUI * scheduler_ui, GtkWidget * vbox) { GtkWidget *combobox; GtkWidget *hbox = add_line(vbox, "Channel: "); combobox = gtk_combo_box_new_text(); scheduler_ui->channel_combobox = combobox; gtk_box_pack_start(GTK_BOX(hbox), combobox, FALSE, FALSE, 0); GMythEPG *mmyth_epg = gmyth_epg_new(); if (!gmyth_epg_connect(mmyth_epg, scheduler_ui->backend_info)) { // FIXME: Without this list the scheduler UI should not be shown! g_warning("[%s] Error when getting list of channels", __FUNCTION__); } if (gmyth_epg_get_channel_list (mmyth_epg, &(scheduler_ui->channel_list)) < 0) { g_debug("[%s] Error while trying to retrieve channel list", __FUNCTION__); } else { GList *clist = scheduler_ui->channel_list; GMythChannelInfo *channel_info; while (clist != NULL) { channel_info = clist->data; clist = clist->next; gtk_combo_box_append_text(GTK_COMBO_BOX (scheduler_ui->channel_combobox), (channel_info->channel_name->str)); } gtk_combo_box_set_active(GTK_COMBO_BOX (scheduler_ui->channel_combobox), 0); } } static void add_time_field(MMythSchedulerUI * scheduler_ui, GtkWidget * vbox) { GtkWidget *label; GtkObject *spinbutton_adj; GtkWidget *hbox = add_line(vbox, "Time: "); time_t real_time; struct tm sched_time; time(&real_time); if (localtime_r((time_t *) & real_time, &sched_time) == NULL) { g_warning("localtime_r error in mmyth_epg_grid_view!\n"); return; } if (sched_time.tm_min > 30) { sched_time.tm_hour = sched_time.tm_hour + 1; sched_time.tm_min = 0; } else if (sched_time.tm_min > 0) { sched_time.tm_min = 30; } scheduler_ui->year_temp = (guint) sched_time.tm_year + 1900; scheduler_ui->month_temp = (guint) sched_time.tm_mon; scheduler_ui->day_temp = (guint) sched_time.tm_mday; // hour entry spinbutton_adj = gtk_adjustment_new(sched_time.tm_hour, 00, 23, 1, 10, 10); scheduler_ui->hour_spinbutton = gtk_spin_button_new(GTK_ADJUSTMENT(spinbutton_adj), 1, 0); gtk_box_pack_start(GTK_BOX(hbox), scheduler_ui->hour_spinbutton, FALSE, FALSE, 0); label = gtk_label_new((" : ")); gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); gtk_label_set_justify(GTK_LABEL(label), GTK_JUSTIFY_RIGHT); gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); // minute entry spinbutton_adj = gtk_adjustment_new(sched_time.tm_min, 0, 59, 1, 10, 10); scheduler_ui->min_spinbutton = gtk_spin_button_new(GTK_ADJUSTMENT(spinbutton_adj), 1, 0); gtk_box_pack_start(GTK_BOX(hbox), scheduler_ui->min_spinbutton, FALSE, FALSE, 0); label = gtk_label_new((" (hh:mm)")); gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); } static void add_date_field(MMythSchedulerUI * scheduler_ui, GtkWidget * vbox) { char sched_date[24]; GtkWidget *hbox = add_line(vbox, "Date: "); // sched_date = ctime(&real_time); g_sprintf(sched_date, "%04d %02d %02d (yyyy mm dd)", scheduler_ui->year_temp, scheduler_ui->month_temp + 1, scheduler_ui->day_temp); sched_date[23] = '\0'; scheduler_ui->date_button = gtk_button_new_with_label(sched_date); gtk_box_pack_start(GTK_BOX(hbox), scheduler_ui->date_button, FALSE, FALSE, 0); gtk_button_set_relief(GTK_BUTTON(scheduler_ui->date_button), GTK_RELIEF_NONE); g_signal_connect(G_OBJECT(scheduler_ui->date_button), "clicked", G_CALLBACK(run_calendar_dialog), scheduler_ui); } static void add_duration_field(MMythSchedulerUI * scheduler_ui, GtkWidget * vbox) { GtkWidget *hbox = add_line(vbox, "Duration: "); GtkWidget *label; GtkObject *spinbutton_adj; spinbutton_adj = gtk_adjustment_new(60, 5, 360, 5, 60, 60); scheduler_ui->duration_spinbutton = gtk_spin_button_new(GTK_ADJUSTMENT(spinbutton_adj), 1, 0); gtk_box_pack_start(GTK_BOX(hbox), scheduler_ui->duration_spinbutton, FALSE, TRUE, 0); gtk_spin_button_set_numeric(GTK_SPIN_BUTTON (scheduler_ui->duration_spinbutton), TRUE); label = gtk_label_new((" (minutes) ")); gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); } static void add_frequency_field(MMythSchedulerUI * scheduler_ui, GtkWidget * vbox) { GtkWidget *hbox = add_line(vbox, "Frequency: "); scheduler_ui->freq_combobox = gtk_combo_box_new_text(); gtk_box_pack_start(GTK_BOX(hbox), scheduler_ui->freq_combobox, FALSE, FALSE, 0); gtk_combo_box_append_text(GTK_COMBO_BOX(scheduler_ui->freq_combobox), ("Only this day ")); gtk_combo_box_append_text(GTK_COMBO_BOX(scheduler_ui->freq_combobox), ("Daily ")); gtk_combo_box_append_text(GTK_COMBO_BOX(scheduler_ui->freq_combobox), ("Weekly ")); gtk_combo_box_set_active(GTK_COMBO_BOX(scheduler_ui->freq_combobox), 0); } static void add_title_field(MMythSchedulerUI * scheduler_ui, GtkWidget * vbox) { GtkWidget *hbox = add_line(vbox, "Title: "); scheduler_ui->title_entry = gtk_entry_new(); gtk_box_pack_start(GTK_BOX(hbox), scheduler_ui->title_entry, FALSE, FALSE, 0); gtk_entry_set_text(GTK_ENTRY(scheduler_ui->title_entry), "(Optional)"); }