Mercurial > audlegacy-plugins
annotate src/alarm/alarm.c @ 1412:6a424e6ff59c
playlist_get_songtime wanted a tuple, but was afraid to ask
author | Tony Vroon <chainsaw@gentoo.org> |
---|---|
date | Mon, 06 Aug 2007 01:37:26 +0100 |
parents | 761e17b23e0c |
children | 1b5f236208d8 |
rev | line source |
---|---|
121 | 1 /* |
2 * Copyright (C) Adam Feakin <adamf@snika.uklinux.net> | |
3 * | |
4 * modified by Daniel Stodden <stodden@in.tum.de> | |
5 * | |
6 * This program is free software; you can redistribute it and/or | |
7 * modify it under the terms of the GNU General Public License | |
8 * as published by the Free Software Foundation; either version 2 | |
9 * of the License, or (at your option) any later version. | |
10 * | |
11 * This program is distributed in the hope that it will be useful, | |
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
14 * GNU General Public License for more details. | |
15 * | |
16 * You should have received a copy of the GNU General Public License | |
17 * along with this program; if not, write to the Free Software | |
18 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. | |
19 */ | |
20 | |
21 /* | |
22 * xmms-alarm.c - a XMMS plugin which allows you to set a time for it to | |
23 * start playing a playlist with the volume fading up and more things | |
24 * the next time I get bored | |
25 */ | |
26 | |
27 /* this file really should get split/cleaned up sometime ;) */ | |
28 #include "config.h" | |
29 | |
30 #if STDC_HEADERS | |
31 # include <stdlib.h> | |
32 #endif | |
33 | |
34 #include <time.h> | |
35 #if TM_IN_SYS_TIME | |
36 # include <sys/time.h> | |
37 #endif | |
38 | |
39 #include <string.h> | |
40 #include <stdio.h> | |
41 #include <audacious/plugin.h> | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
42 #include <audacious/auddrct.h> |
121 | 43 #include <audacious/configdb.h> |
44 #include <gdk/gdk.h> | |
45 #include <gtk/gtk.h> | |
46 #include <pthread.h> | |
47 #include <assert.h> | |
48 #include <math.h> | |
49 | |
50 #include "alarm.h" | |
51 #include "interface.h" | |
52 #include "callbacks.h" | |
53 | |
54 static pthread_t start_tid; /* thread id of alarm loop */ | |
55 static pthread_t stop_tid; /* thread id of stop loop */ | |
56 static pthread_mutex_t fader_lock = PTHREAD_MUTEX_INITIALIZER; | |
57 | |
58 static GeneralPlugin alarm_plugin; | |
59 | |
60 /* string tokens to allow loops and shorten code */ | |
61 static char day_cb[7][7] = {"sun_cb", "mon_cb", "tue_cb", | |
62 "wed_cb", "thu_cb", "fri_cb", "sat_cb"}; | |
63 | |
64 static char day_flags[7][10] = {"sun_flags", "mon_flags", "tue_flags", | |
65 "wed_flags", "thu_flags", "fri_flags", "sat_flags"}; | |
66 | |
67 static char day_h[7][6] = {"sun_h", "mon_h", "tue_h", | |
68 "wed_h", "thu_h", "fri_h", "sat_h"}; | |
69 | |
70 static char day_m[7][6] = {"sun_m", "mon_m", "tue_m", | |
71 "wed_m", "thu_m", "fri_m", "sat_m"}; | |
72 | |
73 static char day_def[7][8] = {"sun_def", "mon_def", "tue_def", | |
74 "wed_def", "thu_def", "fri_def", "sat_def"}; | |
75 | |
76 static struct | |
77 { | |
78 GtkSpinButton *alarm_h; | |
79 GtkSpinButton *alarm_m; | |
80 | |
81 GtkToggleButton *stop_on; | |
82 GtkSpinButton *stop_h; | |
83 GtkSpinButton *stop_m; | |
84 | |
85 GtkRange *volume; | |
86 GtkRange *quietvol; | |
87 | |
88 GtkSpinButton *fading; | |
89 | |
90 GtkEntry *cmdstr; | |
91 GtkToggleButton *cmd_on; | |
92 | |
93 GtkEntry *playlist; | |
94 | |
95 int default_hour; | |
96 int default_min; | |
97 | |
98 // array allows looping of days | |
99 alarmday day[7]; | |
100 | |
101 | |
102 GtkEntry *reminder; | |
103 GtkToggleButton *reminder_cb; | |
104 gchar *reminder_msg; | |
105 gboolean reminder_on; | |
106 } | |
107 alarm_conf; | |
108 | |
109 static gint alarm_h, alarm_m; | |
110 | |
111 static gboolean stop_on; | |
112 static gint stop_h, stop_m; | |
113 | |
114 static gint volume, quietvol; | |
115 | |
116 static gint fading; | |
117 | |
118 static gchar *cmdstr = NULL; | |
119 static gboolean cmd_on; | |
120 | |
121 static gchar *playlist = NULL; | |
122 | |
123 static GtkWidget *config_dialog = NULL; | |
124 static GtkWidget *alarm_dialog = NULL; | |
125 | |
126 static GtkWidget *lookup_widget(GtkWidget *w, const gchar *name) | |
127 { | |
128 GtkWidget *widget; | |
129 | |
130 widget = (GtkWidget*) gtk_object_get_data(GTK_OBJECT(w), | |
131 name); | |
132 g_return_val_if_fail(widget != NULL, NULL); | |
133 | |
134 return widget; | |
135 } | |
136 | |
137 static void dialog_destroyed(GtkWidget *dialog, gpointer data) | |
138 { | |
139 DEBUG("dialog destroyed\n"); | |
140 *(GtkObject**)data = NULL; | |
141 } | |
142 | |
143 static inline gboolean dialog_visible(GtkWidget *dialog) | |
144 { | |
145 return(((dialog != NULL) && GTK_WIDGET_VISIBLE(dialog))); | |
146 } | |
147 | |
148 /* | |
149 * tell the user about that bug | |
150 */ | |
151 static void alarm_warning(void) | |
152 { | |
153 | |
154 static GtkWidget *warning_dialog = NULL; | |
155 | |
156 if(dialog_visible(warning_dialog)) | |
157 return; | |
158 | |
159 warning_dialog = create_warning_dialog(); | |
160 | |
161 gtk_signal_connect(GTK_OBJECT(warning_dialog), "destroy", | |
162 GTK_SIGNAL_FUNC(dialog_destroyed), &warning_dialog); | |
163 | |
164 gtk_widget_show_all(warning_dialog); | |
165 | |
166 return; | |
167 } | |
168 | |
169 /* | |
170 * the callback function that is called when the save button is | |
171 * pressed saves configuration to ~/.bmp/alarmconfig | |
172 */ | |
173 void alarm_save(GtkButton *w, gpointer data) | |
174 { | |
175 int daynum = 0; // used to identify day number | |
176 ConfigDb *conf; | |
177 | |
178 DEBUG("alarm_save\n"); | |
179 | |
180 conf = bmp_cfg_db_open(); | |
181 | |
182 /* | |
183 * update the live values and write them out | |
184 */ | |
185 alarm_h = alarm_conf.default_hour = | |
186 gtk_spin_button_get_value_as_int(alarm_conf.alarm_h); | |
187 bmp_cfg_db_set_int(conf, "alarm", "alarm_h", alarm_h); | |
188 | |
189 alarm_m = alarm_conf.default_min = | |
190 gtk_spin_button_get_value_as_int(alarm_conf.alarm_m); | |
191 bmp_cfg_db_set_int(conf, "alarm", "alarm_m", alarm_m); | |
192 | |
193 | |
194 stop_h = | |
195 gtk_spin_button_get_value_as_int( alarm_conf.stop_h); | |
196 | |
197 stop_m = | |
198 gtk_spin_button_get_value_as_int(alarm_conf.stop_m); | |
199 | |
200 stop_on = | |
201 gtk_toggle_button_get_active(alarm_conf.stop_on); | |
202 | |
203 /* days of the week */ | |
204 for(; daynum < 7; daynum++) | |
205 { | |
206 if(gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(alarm_conf.day[daynum].cb))) | |
207 alarm_conf.day[daynum].flags = 0; | |
208 else | |
209 alarm_conf.day[daynum].flags = ALARM_OFF; | |
210 | |
211 if(gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(alarm_conf.day[daynum].cb_def))) | |
212 alarm_conf.day[daynum].flags |= ALARM_DEFAULT; | |
213 | |
214 alarm_conf.day[daynum].hour = | |
215 gtk_spin_button_get_value_as_int(alarm_conf.day[daynum].spin_hr); | |
216 alarm_conf.day[daynum].min = | |
217 gtk_spin_button_get_value_as_int(alarm_conf.day[daynum].spin_min); | |
218 | |
219 bmp_cfg_db_set_int(conf, "alarm", day_flags[daynum], alarm_conf.day[daynum].flags); | |
220 bmp_cfg_db_set_int(conf, "alarm", day_h[daynum], alarm_conf.day[daynum].hour); | |
221 bmp_cfg_db_set_int(conf, "alarm", day_m[daynum], alarm_conf.day[daynum].min); | |
222 } | |
223 | |
224 /* END: days of week */ | |
225 | |
226 volume = | |
227 gtk_range_get_adjustment(alarm_conf.volume)->value; | |
228 bmp_cfg_db_set_int(conf, "alarm", "volume", volume); | |
229 | |
230 quietvol = | |
231 gtk_range_get_adjustment(alarm_conf.quietvol)->value; | |
232 bmp_cfg_db_set_int(conf, "alarm", "quietvol", quietvol); | |
233 | |
234 fading = | |
235 gtk_spin_button_get_value_as_int(alarm_conf.fading); | |
236 //xmms_cfg_write_int(conf, "alarm", "fading", fading); | |
237 | |
238 /* lets check to see if we need to show the bug warning */ | |
239 if((stop_on == TRUE) && | |
240 ((((stop_h * 60) + stop_m) * 60) < (fading + 65))) | |
241 { | |
242 DEBUG("Displaying bug warning, stop %dh %dm, fade %d\n", | |
243 stop_h, stop_m, fading); | |
244 alarm_warning(); | |
245 } | |
246 else if((stop_on == TRUE) && (fading < 10)) | |
247 { | |
248 DEBUG("Displaying bug warning, stop %dh %dm, fade %d\n", | |
249 stop_h, stop_m, fading); | |
250 alarm_warning(); | |
251 } | |
252 else | |
253 { | |
254 /* write the new values */ | |
255 bmp_cfg_db_set_int(conf, "alarm", "stop_h", stop_h); | |
256 bmp_cfg_db_set_int(conf, "alarm", "stop_m", stop_m); | |
257 bmp_cfg_db_set_int(conf, "alarm", "fading", fading); | |
258 bmp_cfg_db_set_bool(conf, "alarm", "stop_on", stop_on); | |
259 } | |
260 | |
261 | |
262 g_free(cmdstr); | |
263 cmdstr = gtk_editable_get_chars(GTK_EDITABLE(alarm_conf.cmdstr), | |
264 0, -1); | |
265 bmp_cfg_db_set_string(conf, "alarm", "cmdstr", cmdstr); | |
266 | |
267 cmd_on = | |
268 gtk_toggle_button_get_active(alarm_conf.cmd_on); | |
269 bmp_cfg_db_set_bool(conf, "alarm", "cmd_on", cmd_on); | |
270 | |
271 g_free(playlist); | |
272 playlist = gtk_editable_get_chars(GTK_EDITABLE(alarm_conf.playlist), | |
273 0, -1); | |
274 bmp_cfg_db_set_string(conf, "alarm", "playlist", playlist); | |
275 | |
276 /* reminder */ | |
277 g_free(alarm_conf.reminder_msg); | |
278 alarm_conf.reminder_msg = gtk_editable_get_chars(GTK_EDITABLE(alarm_conf.reminder), | |
279 0, -1); | |
280 bmp_cfg_db_set_string(conf, "alarm", "reminder_msg", alarm_conf.reminder_msg); | |
281 | |
282 alarm_conf.reminder_on = | |
283 gtk_toggle_button_get_active(alarm_conf.reminder_cb); | |
284 bmp_cfg_db_set_bool(conf, "alarm", "reminder_on", alarm_conf.reminder_on); | |
285 | |
286 bmp_cfg_db_close(conf); | |
287 } | |
288 | |
289 /* | |
290 * read the current configuration from the file | |
291 */ | |
292 static void alarm_read_config() | |
293 { | |
294 int daynum = 0; // used for day number | |
295 ConfigDb *conf; | |
296 | |
297 DEBUG("alarm_read_config\n"); | |
298 | |
299 conf = bmp_cfg_db_open(); | |
300 | |
301 if(!bmp_cfg_db_get_int(conf, "alarm", "alarm_h", &alarm_h)) | |
302 alarm_h = DEFAULT_ALARM_HOUR; | |
303 if(!bmp_cfg_db_get_int(conf, "alarm", "alarm_m", &alarm_m)) | |
304 alarm_m = DEFAULT_ALARM_MIN; | |
305 | |
306 /* save them here too */ | |
307 alarm_conf.default_hour = alarm_h; | |
308 alarm_conf.default_min = alarm_m; | |
309 | |
310 if(!bmp_cfg_db_get_int( conf, "alarm", "stop_h", &stop_h)) | |
311 stop_h = DEFAULT_STOP_HOURS; | |
312 if(!bmp_cfg_db_get_int( conf, "alarm", "stop_m", &stop_m)) | |
313 stop_m = DEFAULT_STOP_MINS; | |
314 if(!bmp_cfg_db_get_bool(conf, "alarm", "stop_on", &stop_on)) | |
315 stop_on = TRUE; | |
316 | |
317 if(!bmp_cfg_db_get_int(conf, "alarm", "volume", &volume)) | |
318 volume = DEFAULT_VOLUME; | |
319 if(!bmp_cfg_db_get_int(conf, "alarm", "quietvol", &quietvol)) | |
320 quietvol = DEFAULT_QUIET_VOL; | |
321 | |
322 if(!bmp_cfg_db_get_int(conf, "alarm", "fading", &fading)) | |
323 fading = DEFAULT_FADING; | |
324 | |
325 if(!bmp_cfg_db_get_string(conf, "alarm", "cmdstr", &cmdstr)) | |
326 cmdstr = g_strdup(""); | |
327 if(!bmp_cfg_db_get_bool(conf, "alarm", "cmd_on", &cmd_on)) | |
328 cmd_on = FALSE; | |
329 | |
330 if(!bmp_cfg_db_get_string(conf, "alarm", "playlist", &playlist)) | |
331 playlist = g_strdup(""); | |
332 | |
333 if(!bmp_cfg_db_get_string(conf, "alarm", "reminder_msg", &alarm_conf.reminder_msg)) | |
334 alarm_conf.reminder_msg = g_strdup(""); | |
335 if(!bmp_cfg_db_get_bool(conf, "alarm", "reminder_on", &alarm_conf.reminder_on)) | |
336 alarm_conf.reminder_on = FALSE; | |
337 | |
338 /* day flags and times */ | |
339 for(; daynum < 7; daynum++) | |
340 { | |
341 /* read the flags */ | |
342 if(!bmp_cfg_db_get_int(conf, "alarm", day_flags[daynum], &alarm_conf.day[daynum].flags)) { | |
343 // only turn alarm off by default on a sunday | |
344 if(daynum != 0) | |
345 alarm_conf.day[daynum].flags = DEFAULT_FLAGS; | |
346 else | |
347 alarm_conf.day[daynum].flags = DEFAULT_FLAGS | ALARM_OFF; | |
348 } | |
349 | |
350 /* read the times */ | |
351 if(!bmp_cfg_db_get_int(conf, "alarm", day_h[daynum], &alarm_conf.day[daynum].hour)) | |
352 alarm_conf.day[daynum].hour = DEFAULT_ALARM_HOUR; | |
353 | |
354 if(!bmp_cfg_db_get_int(conf, "alarm", day_m[daynum], &alarm_conf.day[daynum].min)) | |
355 alarm_conf.day[daynum].min = DEFAULT_ALARM_MIN; | |
356 } | |
357 | |
358 DEBUG("END alarm_read_config\n"); | |
359 } | |
360 | |
361 /* | |
362 * display an about box | |
363 */ | |
364 static void alarm_about() | |
365 { | |
366 static GtkWidget *about_dialog = NULL; | |
367 | |
368 DEBUG("alarm_about\n"); | |
369 | |
370 if(dialog_visible(about_dialog)) | |
371 return; | |
372 | |
373 about_dialog = create_about_dialog(); | |
374 | |
375 gtk_signal_connect(GTK_OBJECT(about_dialog), "destroy", | |
376 GTK_SIGNAL_FUNC(dialog_destroyed), &about_dialog); | |
377 | |
378 gtk_widget_show_all(about_dialog); | |
379 | |
380 return; | |
381 } | |
382 | |
383 /* | |
384 * create a playlist file selection dialog | |
385 */ | |
386 static void alarm_playlist_browse(GtkButton *button, gpointer data) | |
387 { | |
388 GtkWidget *fs; | |
389 gchar *dirname, *path; | |
390 | |
391 dirname = g_dirname(playlist); | |
392 DEBUG("dirname = %s\n", dirname); | |
393 path = g_strdup_printf("%s/", dirname); | |
394 DEBUG("path = %s\n", path); | |
395 g_free(dirname); | |
396 | |
397 fs = create_playlist_fileselection(); | |
398 | |
399 gtk_file_selection_set_filename(GTK_FILE_SELECTION(fs), path); | |
400 g_free(path); | |
401 | |
402 gtk_widget_show_all(fs); | |
403 } | |
404 | |
405 /* | |
406 * save selected playlist to the corresponding text entry | |
407 */ | |
408 void alarm_store_playlistname(GtkButton *button, gpointer data) | |
409 { | |
410 GtkFileSelection *fs = GTK_FILE_SELECTION(data); | |
411 gchar *plist; | |
412 | |
413 DEBUG("alarm_store_playlistname\n"); | |
414 | |
132
26a5aef73955
[svn] Fixes compiler warnings that "Joker" was talking about in #audacious.
asheldon
parents:
129
diff
changeset
|
415 plist = g_strdup(gtk_file_selection_get_filename(fs)); |
121 | 416 |
417 gtk_entry_set_text(alarm_conf.playlist, plist); | |
132
26a5aef73955
[svn] Fixes compiler warnings that "Joker" was talking about in #audacious.
asheldon
parents:
129
diff
changeset
|
418 g_free(plist); |
121 | 419 } |
420 | |
421 /* | |
422 * displays the configuration window and opens the config file. | |
423 */ | |
424 static void alarm_configure(void) | |
425 { | |
426 int daynum = 0; // used to loop days | |
427 GtkWidget *w; | |
428 | |
429 DEBUG("alarm_configure\n"); | |
430 | |
431 /* | |
432 * dont want to show more than one config window | |
433 */ | |
434 if(dialog_visible(config_dialog)) | |
435 return; | |
436 | |
437 alarm_read_config(); | |
438 | |
439 /* | |
440 * Create the widgets | |
441 */ | |
442 config_dialog = create_config_dialog(); | |
443 | |
444 w = lookup_widget(config_dialog, "alarm_h_spin"); | |
445 alarm_conf.alarm_h = GTK_SPIN_BUTTON(w); | |
446 gtk_spin_button_set_value(alarm_conf.alarm_h, alarm_h); | |
447 | |
448 w = lookup_widget(config_dialog, "alarm_m_spin"); | |
449 alarm_conf.alarm_m = GTK_SPIN_BUTTON(w); | |
450 gtk_spin_button_set_value(alarm_conf.alarm_m, alarm_m); | |
451 | |
452 w = lookup_widget(config_dialog, "stop_h_spin"); | |
453 alarm_conf.stop_h = GTK_SPIN_BUTTON(w); | |
454 gtk_spin_button_set_value(alarm_conf.stop_h, stop_h); | |
455 | |
456 w = lookup_widget(config_dialog, "stop_m_spin"); | |
457 alarm_conf.stop_m = GTK_SPIN_BUTTON(w); | |
458 gtk_spin_button_set_value(alarm_conf.stop_m, stop_m); | |
459 | |
460 w = lookup_widget(config_dialog, "stop_checkb"); | |
461 alarm_conf.stop_on = GTK_TOGGLE_BUTTON(w); | |
462 gtk_toggle_button_set_active(alarm_conf.stop_on, stop_on); | |
463 | |
464 w = lookup_widget(config_dialog, "vol_scale"); | |
465 alarm_conf.volume = GTK_RANGE(w); | |
466 gtk_range_set_adjustment(alarm_conf.volume, | |
467 GTK_ADJUSTMENT(gtk_adjustment_new(volume, | |
468 0, | |
469 100, 1, | |
470 5, 0))); | |
471 | |
472 w = lookup_widget(config_dialog, "quiet_vol_scale"); | |
473 alarm_conf.quietvol = GTK_RANGE(w); | |
474 gtk_range_set_adjustment(alarm_conf.quietvol, | |
475 GTK_ADJUSTMENT(gtk_adjustment_new(quietvol, | |
476 0, | |
477 100, 1, | |
478 5, 0))); | |
479 | |
480 /* days of week */ | |
481 for(; daynum < 7; daynum++) | |
482 { | |
483 w = lookup_widget(config_dialog, day_cb[daynum]); | |
484 alarm_conf.day[daynum].cb = GTK_CHECK_BUTTON(w); | |
485 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(alarm_conf.day[daynum].cb), | |
486 !(alarm_conf.day[daynum].flags & ALARM_OFF)); | |
487 | |
488 w = lookup_widget(config_dialog, day_def[daynum]); | |
489 alarm_conf.day[daynum].cb_def = GTK_CHECK_BUTTON(w); | |
490 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(alarm_conf.day[daynum].cb_def), | |
491 alarm_conf.day[daynum].flags & ALARM_DEFAULT); | |
492 | |
493 | |
494 /* Changed to show default time instead of set time when ALARM_DEFAULT set, | |
495 * as suggested by Mark Brown | |
496 */ | |
497 /* w = lookup_widget(config_dialog, day_h[daynum]); | |
498 alarm_conf.day[daynum].spin_hr = GTK_SPIN_BUTTON(w); | |
499 gtk_spin_button_set_value(alarm_conf.day[daynum].spin_hr, alarm_conf.day[daynum].hour); | |
500 | |
501 w = lookup_widget(config_dialog, day_m[daynum]); | |
502 alarm_conf.day[daynum].spin_min = GTK_SPIN_BUTTON(w); | |
503 gtk_spin_button_set_value(alarm_conf.day[daynum].spin_min, alarm_conf.day[daynum].min); | |
504 */ | |
505 if(alarm_conf.day[daynum].flags & ALARM_DEFAULT) | |
506 { | |
507 w = lookup_widget(config_dialog, day_h[daynum]); | |
508 alarm_conf.day[daynum].spin_hr = GTK_SPIN_BUTTON(w); | |
509 gtk_spin_button_set_value(alarm_conf.day[daynum].spin_hr, alarm_conf.default_hour); | |
510 | |
511 w = lookup_widget(config_dialog, day_m[daynum]); | |
512 alarm_conf.day[daynum].spin_min = GTK_SPIN_BUTTON(w); | |
513 gtk_spin_button_set_value(alarm_conf.day[daynum].spin_min, alarm_conf.default_min); | |
514 | |
515 gtk_widget_set_sensitive((GtkWidget *)alarm_conf.day[daynum].spin_hr, FALSE); | |
516 gtk_widget_set_sensitive((GtkWidget *)alarm_conf.day[daynum].spin_min, FALSE); | |
517 } | |
518 else | |
519 { | |
520 w = lookup_widget(config_dialog, day_h[daynum]); | |
521 alarm_conf.day[daynum].spin_hr = GTK_SPIN_BUTTON(w); | |
522 gtk_spin_button_set_value(alarm_conf.day[daynum].spin_hr, alarm_conf.day[daynum].hour); | |
523 | |
524 w = lookup_widget(config_dialog, day_m[daynum]); | |
525 alarm_conf.day[daynum].spin_min = GTK_SPIN_BUTTON(w); | |
526 gtk_spin_button_set_value(alarm_conf.day[daynum].spin_min, alarm_conf.day[daynum].min); | |
527 | |
528 gtk_widget_set_sensitive((GtkWidget *)alarm_conf.day[daynum].spin_hr, TRUE); | |
529 gtk_widget_set_sensitive((GtkWidget *)alarm_conf.day[daynum].spin_min, TRUE); | |
530 } | |
531 } | |
532 | |
533 /* END: days of week */ | |
534 | |
535 w = lookup_widget(config_dialog,"fading_spin"); | |
536 alarm_conf.fading = GTK_SPIN_BUTTON(w); | |
537 gtk_spin_button_set_value(alarm_conf.fading, fading); | |
538 | |
539 w = lookup_widget(config_dialog, "cmd_entry"); | |
540 alarm_conf.cmdstr = GTK_ENTRY(w); | |
541 gtk_entry_set_text(alarm_conf.cmdstr, cmdstr); | |
542 | |
543 w = lookup_widget(config_dialog, "cmd_checkb"); | |
544 alarm_conf.cmd_on = GTK_TOGGLE_BUTTON(w); | |
545 gtk_toggle_button_set_active(alarm_conf.cmd_on, cmd_on); | |
546 | |
547 w = lookup_widget(config_dialog, "playlist"); | |
548 alarm_conf.playlist = GTK_ENTRY(w); | |
549 gtk_entry_set_text(alarm_conf.playlist, playlist); | |
550 | |
551 w = lookup_widget(config_dialog, "reminder_text"); | |
552 alarm_conf.reminder = GTK_ENTRY(w); | |
553 gtk_entry_set_text(alarm_conf.reminder, alarm_conf.reminder_msg); | |
554 | |
555 w = lookup_widget(config_dialog, "reminder_cb"); | |
556 alarm_conf.reminder_cb = GTK_TOGGLE_BUTTON(w); | |
557 gtk_toggle_button_set_active(alarm_conf.reminder_cb, alarm_conf.reminder_on); | |
558 | |
559 w = lookup_widget(config_dialog, "playlist_browse_button"); | |
560 gtk_signal_connect(GTK_OBJECT(w), "clicked", | |
561 GTK_SIGNAL_FUNC(alarm_playlist_browse), NULL); | |
562 | |
563 gtk_signal_connect(GTK_OBJECT(config_dialog), "destroy", | |
564 GTK_SIGNAL_FUNC(dialog_destroyed), &config_dialog); | |
565 | |
566 gtk_widget_show_all(config_dialog); | |
567 | |
568 DEBUG("END alarm_configure\n"); | |
569 } | |
570 | |
571 /* functions for greying out the time for days */ | |
572 void on_day_def_toggled(GtkToggleButton *togglebutton, gpointer user_data, int daynum) | |
573 { | |
574 GtkWidget *w; | |
575 | |
576 /* change the time shown too */ | |
577 w = lookup_widget(config_dialog, day_h[daynum]); | |
578 if(w == NULL) | |
579 return; | |
580 | |
581 if(gtk_toggle_button_get_active(togglebutton) == TRUE) | |
582 { | |
583 gtk_spin_button_set_value(GTK_SPIN_BUTTON(w), alarm_conf.default_hour); | |
584 gtk_widget_set_sensitive(w, FALSE); | |
585 } | |
586 else | |
587 { | |
588 gtk_spin_button_set_value(GTK_SPIN_BUTTON(w), alarm_conf.day[daynum].hour); | |
589 gtk_widget_set_sensitive(w, TRUE); | |
590 } | |
591 | |
592 w = lookup_widget(config_dialog, day_m[daynum]); | |
593 if(gtk_toggle_button_get_active(togglebutton) == TRUE) | |
594 { | |
595 gtk_spin_button_set_value(GTK_SPIN_BUTTON(w), alarm_conf.default_min); | |
596 gtk_widget_set_sensitive(w, FALSE); | |
597 } | |
598 else | |
599 { | |
600 gtk_spin_button_set_value(GTK_SPIN_BUTTON(w), alarm_conf.day[daynum].min); | |
601 gtk_widget_set_sensitive(w, TRUE); | |
602 } | |
603 } | |
604 | |
605 void on_sun_def_toggled(GtkToggleButton *togglebutton, gpointer user_data) | |
606 { | |
607 on_day_def_toggled(togglebutton, user_data, 0); | |
608 } | |
609 | |
610 void on_mon_def_toggled(GtkToggleButton *togglebutton, gpointer user_data) | |
611 { | |
612 on_day_def_toggled(togglebutton, user_data, 1); | |
613 } | |
614 | |
615 void on_tue_def_toggled(GtkToggleButton *togglebutton, gpointer user_data) | |
616 { | |
617 on_day_def_toggled(togglebutton, user_data, 2); | |
618 } | |
619 | |
620 void on_wed_def_toggled(GtkToggleButton *togglebutton, gpointer user_data) | |
621 { | |
622 on_day_def_toggled(togglebutton, user_data, 3); | |
623 } | |
624 | |
625 void on_thu_def_toggled(GtkToggleButton *togglebutton, gpointer user_data) | |
626 { | |
627 on_day_def_toggled(togglebutton, user_data, 4); | |
628 } | |
629 | |
630 void on_fri_def_toggled(GtkToggleButton *togglebutton, gpointer user_data) | |
631 { | |
632 on_day_def_toggled(togglebutton, user_data, 5); | |
633 } | |
634 | |
635 void on_sat_def_toggled(GtkToggleButton *togglebutton, gpointer user_data) | |
636 { | |
637 on_day_def_toggled(togglebutton, user_data, 6); | |
638 } | |
639 | |
640 /* END: greying things */ | |
641 | |
642 void alarm_current_volume(GtkButton *button, gpointer data) | |
643 { | |
644 gint vol; | |
645 GtkAdjustment *adj; | |
646 | |
647 DEBUG("on_current_button_clicked\n"); | |
648 | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
649 audacious_drct_get_volume_main(&vol); |
121 | 650 |
651 adj = gtk_range_get_adjustment(alarm_conf.volume); | |
652 gtk_adjustment_set_value(adj, (gfloat)vol); | |
653 } | |
654 | |
655 /* | |
656 * a thread safe sleeping function - | |
657 * and it even works in solaris (I think) | |
658 */ | |
659 static void threadsleep(float x) | |
660 { | |
661 DEBUG("threadsleep: waiting %f seconds\n", x); | |
662 | |
122
96fc1ef32c99
[svn] - push fixes from p4://depot/svn-audacious-plugins-devel/virt-branches/audalarm/src
nenolod
parents:
121
diff
changeset
|
663 g_usleep((int) ((float) x * (float) 1000000.0)); |
121 | 664 |
665 return; | |
666 } | |
667 | |
668 static inline pthread_t alarm_thread_create(void *(*start_routine)(void *), void *args, unsigned int detach) | |
669 { | |
670 pthread_t tid; | |
671 pthread_attr_t attr; | |
672 | |
673 pthread_attr_init(&attr); | |
674 | |
675 if(detach != 0) | |
676 pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED); | |
677 | |
678 pthread_attr_setinheritsched(&attr, PTHREAD_EXPLICIT_SCHED); | |
679 pthread_attr_setschedpolicy(&attr, SCHED_OTHER); | |
680 pthread_attr_setscope(&attr, PTHREAD_SCOPE_SYSTEM); | |
681 | |
682 pthread_create(&tid, &attr, start_routine, args); | |
683 | |
684 return(tid); | |
685 } | |
686 | |
687 static void *alarm_fade(void *arg) | |
688 { | |
689 fader *vols = (fader *)arg; | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
690 guint i, v; |
121 | 691 gint inc, diff, adiff; |
692 | |
693 /* lock */ | |
694 pthread_mutex_lock(&fader_lock); | |
695 | |
696 /* slide volume */ | |
697 /* the Kaspar Giger way of fading, check the current mixer volume and | |
698 * increment from there so that if you have some other app lowering the | |
699 * volume at the same time xmms-alarm will not ignore it. If you have some | |
700 * other app increasing the volume, then it could get louder that you expect | |
701 * though - because the loop does not recalculate the difference each time. | |
702 */ | |
703 | |
704 /* difference between the 2 volumes */ | |
705 diff = vols->end - vols->start; | |
706 adiff = abs(diff); | |
707 | |
708 /* Are we going up or down? */ | |
709 if(diff < 0) | |
710 inc = -1; | |
711 else | |
712 inc = 1; | |
713 | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
714 audacious_drct_set_volume_main((gint)vols->start); |
121 | 715 //for(i=0;i<(vols->end - vols->start);i++) |
716 for(i=0;i<adiff;i++) | |
717 { | |
718 //threadsleep((gfloat)fading / (vols->end - vols->start)); | |
719 threadsleep((gfloat)fading / (gfloat)adiff); | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
720 audacious_drct_get_volume_main(&v); |
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
721 audacious_drct_set_volume_main((gint)(v + inc)); |
121 | 722 } |
723 /* Setting the volume to the end volume sort of defeats the point if having | |
724 * the code in there to allow other apps to control volume too :) | |
725 */ | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
726 //audacious_drct_set_volume_main((gint)vols->end); |
121 | 727 |
728 /* and */ | |
729 pthread_mutex_unlock(&fader_lock); | |
730 | |
731 DEBUG("volume = %f%%\n", (gdouble)vols->end); | |
732 return(0); | |
733 } | |
734 | |
735 static void *alarm_stop_thread( void *args ) | |
736 { | |
737 gint currvol; | |
738 fader fade_vols; | |
739 pthread_t f_tid; | |
740 | |
741 DEBUG("alarm_stop_thread\n"); | |
742 | |
743 | |
744 /* sleep for however long we are meant to be sleeping for until | |
745 * its time to shut up | |
746 */ | |
747 threadsleep(((stop_h * 60) + stop_m) * 60); | |
748 | |
749 DEBUG("alarm_stop triggered\n"); | |
750 | |
751 if (dialog_visible(alarm_dialog)) | |
752 gtk_widget_destroy(alarm_dialog); | |
753 | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
754 audacious_drct_get_volume_main(&currvol), |
121 | 755 |
756 /* fade back to zero */ | |
757 fade_vols.start = currvol; | |
758 fade_vols.end = 0; | |
759 | |
760 /* The fader thread locks the fader_mutex now */ | |
761 f_tid = alarm_thread_create(alarm_fade, &fade_vols, 0); | |
762 | |
763 pthread_join(f_tid, NULL); | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
764 audacious_drct_stop(); |
121 | 765 |
766 /* might as well set the volume to something higher than zero so we | |
767 * dont confuse the poor people who just woke up and cant work out why | |
768 * theres no music playing when they press the little play button :) | |
769 */ | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
770 audacious_drct_set_volume_main(currvol); |
121 | 771 |
772 DEBUG("alarm_stop done\n"); | |
773 return(NULL); | |
774 } | |
775 | |
776 void alarm_stop_cancel(GtkButton *w, gpointer data) | |
777 { | |
778 DEBUG("alarm_stop_cancel\n"); | |
779 pthread_cancel(stop_tid); | |
780 } | |
781 | |
782 /* the main alarm thread */ | |
783 static void *alarm_start_thread(void *args) | |
784 { | |
785 struct tm *currtime; | |
786 time_t timenow; | |
787 unsigned int play_start = 0; | |
788 guint today; | |
789 | |
790 /* give it time to set start_tid to something */ | |
791 threadsleep(1); | |
792 | |
793 while(start_tid != 0) | |
794 { | |
795 /* sit around and wait for the faders to not be doing anything */ | |
796 DEBUG("Waiting for fader to be unlocked.."); | |
797 pthread_mutex_lock(&fader_lock); | |
798 DEBUG("Ok\n"); | |
799 pthread_mutex_unlock(&fader_lock); | |
800 | |
801 DEBUG("Getting time\n"); | |
802 timenow = time(NULL); | |
803 currtime = localtime(&timenow); | |
804 today = currtime->tm_wday; | |
805 DEBUG("Today is %d\n", today); | |
806 | |
807 /* see if its time to do something */ | |
808 DEBUG("Checking Day\n"); | |
809 | |
810 /* Had to put something here so I put the hour string. | |
811 ** Its only debug stuff anyway */ | |
812 DEBUG(day_h[today]); | |
813 | |
814 if(alarm_conf.day[today].flags & ALARM_OFF) | |
815 { | |
816 threadsleep(8.5); | |
817 continue; | |
818 } | |
819 else | |
820 { | |
821 /* set the alarm_h and alarm_m for today, if not default */ | |
822 if(!(alarm_conf.day[today].flags & ALARM_DEFAULT)) | |
823 { | |
824 alarm_h = alarm_conf.day[today].hour; | |
825 alarm_m = alarm_conf.day[today].min; | |
826 } | |
827 else | |
828 { | |
829 alarm_h = alarm_conf.default_hour; | |
830 alarm_m = alarm_conf.default_min; | |
831 } | |
832 } | |
833 | |
834 DEBUG("Alarm time is %d:%d (def: %d:%d)\n", alarm_h, alarm_m, | |
835 alarm_conf.default_hour, alarm_conf.default_min); | |
836 | |
837 DEBUG("Checking time (%d:%d)\n", currtime->tm_hour, currtime->tm_min); | |
838 if((currtime->tm_hour != alarm_h) || (currtime->tm_min != alarm_m)) | |
839 { | |
840 threadsleep(8.5); | |
841 continue; | |
842 } | |
843 | |
844 if(cmd_on == TRUE) | |
845 { | |
846 DEBUG("Executing %s, cmd_on is true\n", cmdstr); | |
963 | 847 if(system(cmdstr) == -1) |
848 { | |
849 DEBUG("Executing %s failed\n",cmdstr); | |
850 } | |
121 | 851 } |
852 | |
853 DEBUG("strcmp playlist, playlist is [%s]\n", playlist); | |
854 if(strcmp(playlist, "")) | |
855 { | |
856 DEBUG("playlist is not blank, aparently\n"); | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
857 GList list; |
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
858 |
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
859 list.prev = list.next = NULL; |
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
860 list.data = playlist; |
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
861 |
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
862 audacious_drct_pl_clear(); |
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
863 audacious_drct_pl_add(&list); |
121 | 864 } |
865 | |
866 if(fading) | |
867 { | |
868 fader fade_vols; | |
869 | |
870 DEBUG("Fading is true\n"); | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
871 audacious_drct_set_volume_main(quietvol); |
121 | 872 |
873 /* start playing */ | |
874 play_start = time(NULL); | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
875 audacious_drct_play(); |
121 | 876 |
877 /* fade volume */ | |
878 fade_vols.start = quietvol; | |
879 fade_vols.end = volume; | |
880 | |
881 //alarm_fade(quietvol, volume); | |
882 alarm_thread_create(alarm_fade, &fade_vols, 0); | |
883 } | |
884 else | |
885 { | |
886 /* no fading */ | |
887 | |
888 /* set volume */ | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
889 audacious_drct_set_volume_main(volume); |
121 | 890 |
891 /* start playing */ | |
892 play_start = time(NULL); | |
1035
711ec8d39ca6
[svn] Changed alarm and projectm plugins so they use auddrct now.
magma
parents:
1005
diff
changeset
|
893 audacious_drct_play(); |
121 | 894 } |
895 | |
896 if(alarm_conf.reminder_on == TRUE) | |
897 { | |
898 GtkWidget *reminder_dialog; | |
899 DEBUG("Showing reminder '%s'\n", alarm_conf.reminder_msg); | |
900 | |
901 GDK_THREADS_ENTER(); | |
132
26a5aef73955
[svn] Fixes compiler warnings that "Joker" was talking about in #audacious.
asheldon
parents:
129
diff
changeset
|
902 reminder_dialog = (GtkWidget*) create_reminder_dialog(alarm_conf.reminder_msg); |
121 | 903 gtk_signal_connect(GTK_OBJECT(reminder_dialog), "destroy", |
904 GTK_SIGNAL_FUNC(dialog_destroyed), &reminder_dialog); | |
905 gtk_widget_show_all(reminder_dialog); | |
906 GDK_THREADS_LEAVE(); | |
907 } | |
908 | |
909 /* bring up the wakeup call dialog if stop_on is set TRUE, this | |
910 * has been moved to after making xmms play so that it doesnt | |
911 * get in the way for people with manual window placement turned on | |
912 * | |
913 * this means that the dialog doesnt get shown until the volume has | |
914 * finished fading though !, so thats something else to fix | |
915 */ | |
916 if(stop_on == TRUE) | |
917 { | |
918 /* ok, so when we want to open dialogs in threaded programs | |
919 * we use this do we? | |
920 * anyone? | |
921 */ | |
922 GDK_THREADS_ENTER(); | |
923 { | |
924 DEBUG("stop_on is true\n"); | |
925 alarm_dialog = create_alarm_dialog(); | |
926 DEBUG("created alarm dialog, %p\n", alarm_dialog); | |
927 | |
928 gtk_signal_connect(GTK_OBJECT(alarm_dialog), "destroy", | |
929 GTK_SIGNAL_FUNC(dialog_destroyed), &alarm_dialog); | |
930 DEBUG("attached destroy signal to alarm dialog, %p\n", alarm_dialog); | |
931 gtk_widget_show_all(alarm_dialog); | |
932 DEBUG("dialog now showing\n"); | |
933 | |
934 DEBUG("now starting stop thread\n"); | |
935 stop_tid = alarm_thread_create(alarm_stop_thread, NULL, 0); | |
936 DEBUG("Created wakeup dialog and started stop thread(%d)\n", (int)stop_tid); | |
937 | |
938 } | |
939 GDK_THREADS_LEAVE(); | |
940 | |
941 /* now wait for the stop thread */ | |
942 DEBUG("Waiting for stop to stop.... (%d)", (int)stop_tid); | |
943 pthread_join(stop_tid, NULL); | |
944 /* loop until we are out of the starting minute */ | |
945 while(time(NULL) < (play_start + 61)) | |
946 { | |
947 DEBUG("Waiting until out of starting minute\n"); | |
948 threadsleep(5.0); | |
949 } | |
950 DEBUG("OK\n"); | |
951 } | |
952 /* loop until we are out of the starting minute */ | |
953 while(time(NULL) < (play_start + 61)) | |
954 { | |
955 threadsleep(5.0); | |
956 } | |
957 threadsleep(fading); | |
958 } | |
959 | |
960 DEBUG("Main thread has gone...\n"); | |
961 return NULL; | |
962 } | |
963 | |
964 /* | |
965 * initialization | |
966 * opens the config file and reads the value, creates a new | |
967 * config in memory if the file doesnt exist and sets default vals | |
968 */ | |
969 static void alarm_init() | |
970 { | |
971 DEBUG("alarm_init\n"); | |
972 | |
973 alarm_read_config(); | |
974 | |
975 /* start the main thread running */ | |
976 start_tid = alarm_thread_create(alarm_start_thread, NULL, 1); | |
977 } | |
978 | |
979 /* | |
980 * kill the main thread | |
981 */ | |
982 static void alarm_cleanup() | |
983 { | |
984 DEBUG("alarm_cleanup\n"); | |
985 | |
129
dae03298753f
[svn] - don't cancel threads that don't exist (TODO: GThreadify)
nenolod
parents:
122
diff
changeset
|
986 if (start_tid) |
dae03298753f
[svn] - don't cancel threads that don't exist (TODO: GThreadify)
nenolod
parents:
122
diff
changeset
|
987 pthread_cancel(start_tid); |
121 | 988 start_tid = 0; |
989 if(stop_tid) | |
990 pthread_cancel(stop_tid); | |
129
dae03298753f
[svn] - don't cancel threads that don't exist (TODO: GThreadify)
nenolod
parents:
122
diff
changeset
|
991 stop_tid = 0; |
121 | 992 } |
993 | |
994 /* | |
995 * for xmms to get the function names | |
996 */ | |
997 static GeneralPlugin alarm_plugin = | |
998 { | |
999 NULL, | |
1060
2649ba83333b
[svn] - remove xmms_session value from GeneralPlugin structs per plugin api v2
nenolod
parents:
1035
diff
changeset
|
1000 NULL, |
121 | 1001 "Alarm "VERSION, |
1002 alarm_init, | |
1003 alarm_about, | |
1004 alarm_configure, | |
1005 alarm_cleanup, | |
1006 }; | |
1007 | |
1120 | 1008 GeneralPlugin *alarm_gplist[] = { &alarm_plugin, NULL }; |
1395
761e17b23e0c
added Discovery plugin type
Cristi Magherusan <majeru@atheme-project.org>
parents:
1120
diff
changeset
|
1009 DECLARE_PLUGIN(alarm, NULL, NULL, NULL, NULL, NULL, alarm_gplist, NULL, NULL); |
1120 | 1010 |
121 | 1011 /* |
1012 * vi:ai:expandtab:ts=2 sts=2 shiftwidth=2:nowrap: | |
1013 */ |