Mercurial > audlegacy
annotate Plugins/Output/disk_writer/disk_writer.c @ 1674:62ee0900bb9a trunk
[svn] Remove unused bmp_player.xpm; as requested by Mikachu on #audacious.
author | chainsaw |
---|---|
date | Wed, 13 Sep 2006 05:22:45 -0700 |
parents | de0a0927a5c2 |
children |
rev | line source |
---|---|
61 | 1 /* XMMS - Cross-platform multimedia player |
2 * Copyright (C) 1998-2004 Peter Alm, Mikael Alm, Olle Hallnas, | |
3 * Thomas Nilsson and 4Front Technologies | |
4 * Copyright (C) 1999-2004 Haavard Kvaalen | |
5 * | |
6 * File name suffix option added by Heikki Orsila 2003 | |
7 * <heikki.orsila@iki.fi> (no copyrights claimed) | |
8 * | |
9 * This program is free software; you can redistribute it and/or modify | |
10 * it under the terms of the GNU General Public License as published by | |
11 * the Free Software Foundation; either version 2 of the License, or | |
12 * (at your option) any later version. | |
13 * | |
14 * This program is distributed in the hope that it will be useful, | |
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
17 * GNU General Public License for more details. | |
18 * | |
19 * You should have received a copy of the GNU General Public License | |
20 * along with this program; if not, write to the Free Software | |
1459 | 21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. |
61 | 22 */ |
23 | |
24 #include "config.h" | |
25 | |
26 #include <gtk/gtk.h> | |
27 #include <glib/gi18n.h> | |
28 #include <stdio.h> | |
29 #include <string.h> | |
30 | |
31 #include "audacious/plugin.h" | |
32 #include "libaudacious/beepctrl.h" | |
33 #include "libaudacious/dirbrowser.h" | |
311 | 34 #include "libaudacious/configdb.h" |
61 | 35 #include "libaudacious/util.h" |
1664 | 36 #include "libaudacious/vfs.h" |
61 | 37 |
38 struct wavhead | |
39 { | |
40 guint32 main_chunk; | |
41 guint32 length; | |
42 guint32 chunk_type; | |
43 guint32 sub_chunk; | |
44 guint32 sc_len; | |
45 guint16 format; | |
46 guint16 modus; | |
47 guint32 sample_fq; | |
48 guint32 byte_p_sec; | |
49 guint16 byte_p_spl; | |
50 guint16 bit_p_spl; | |
51 guint32 data_chunk; | |
52 guint32 data_length; | |
53 }; | |
54 | |
55 static GtkWidget *configure_win = NULL, *configure_vbox; | |
56 static GtkWidget *path_hbox, *path_label, *path_entry, *path_browse, *path_dirbrowser = NULL; | |
57 static GtkWidget *configure_separator; | |
58 static GtkWidget *configure_bbox, *configure_ok, *configure_cancel; | |
59 | |
60 static GtkWidget *use_suffix_toggle = NULL; | |
61 static gboolean use_suffix = FALSE; | |
62 | |
63 static gchar *file_path = NULL; | |
1664 | 64 static VFSFile *output_file = NULL; |
61 | 65 static struct wavhead header; |
66 static guint64 written = 0; | |
67 static AFormat afmt; | |
68 gint ctrlsocket_get_session_id(void); /* FIXME */ | |
69 | |
70 static void disk_init(void); | |
71 static gint disk_open(AFormat fmt, gint rate, gint nch); | |
72 static void disk_write(void *ptr, gint length); | |
73 static void disk_close(void); | |
74 static void disk_flush(gint time); | |
75 static void disk_pause(short p); | |
76 static gint disk_free(void); | |
77 static gint disk_playing(void); | |
78 static gint disk_get_written_time(void); | |
79 static gint disk_get_output_time(void); | |
80 static void disk_configure(void); | |
543 | 81 static void disk_getvol(gint *, gint *); |
82 static void disk_setvol(gint, gint); | |
83 | |
84 static int lvol = 0, rvol = 0; | |
61 | 85 |
86 OutputPlugin disk_op = | |
87 { | |
88 NULL, | |
89 NULL, | |
90 NULL, /* Description */ | |
91 disk_init, | |
309 | 92 NULL, |
61 | 93 NULL, /* about */ |
94 disk_configure, /* configure */ | |
543 | 95 disk_getvol, /* get_volume */ |
96 disk_setvol, /* set_volume */ | |
61 | 97 disk_open, |
98 disk_write, | |
99 disk_close, | |
100 disk_flush, | |
101 disk_pause, | |
102 disk_free, | |
103 disk_playing, | |
104 disk_get_output_time, | |
105 disk_get_written_time, | |
517
55dc40ff1aff
[svn] Add tell_audio(AFormat *, gint *, gint *), for retrieving audio
nenolod
parents:
311
diff
changeset
|
106 NULL |
61 | 107 }; |
108 | |
109 OutputPlugin *get_oplugin_info(void) | |
110 { | |
111 disk_op.description = g_strdup_printf(_("Disk Writer Plugin %s"), VERSION); | |
112 return &disk_op; | |
113 } | |
114 | |
115 static void disk_init(void) | |
116 { | |
311 | 117 ConfigDb *db; |
61 | 118 |
311 | 119 db = bmp_cfg_db_open(); |
120 bmp_cfg_db_get_string(db, "disk_writer", "file_path", &file_path); | |
121 bmp_cfg_db_get_bool(db, "disk_writer", "use_suffix", &use_suffix); | |
122 bmp_cfg_db_close(db); | |
61 | 123 |
124 if (!file_path) | |
125 file_path = g_strdup(g_get_home_dir()); | |
126 } | |
127 | |
128 static gint disk_open(AFormat fmt, gint rate, gint nch) | |
129 { | |
130 gchar *filename, *title, *temp; | |
131 gint pos; | |
132 | |
133 written = 0; | |
134 afmt = fmt; | |
135 | |
136 if (xmms_check_realtime_priority()) | |
137 { | |
138 xmms_show_message(_("Error"), | |
139 _("You cannot use the Disk Writer plugin\n" | |
140 "when you're running in realtime mode."), | |
141 _("OK"), FALSE, NULL, NULL); | |
142 return 0; | |
143 } | |
144 | |
145 pos = xmms_remote_get_playlist_pos(ctrlsocket_get_session_id()); | |
146 title = xmms_remote_get_playlist_file(ctrlsocket_get_session_id(), pos); | |
147 if (!use_suffix) { | |
148 if (title != NULL && (temp = strrchr(title, '.')) != NULL) { | |
149 *temp = '\0'; | |
150 } | |
151 } | |
152 if (title == NULL || strlen(g_basename(title)) == 0) | |
153 { | |
154 g_free(title); | |
155 /* No filename, lets try title instead */ | |
156 title = xmms_remote_get_playlist_title(ctrlsocket_get_session_id(), pos); | |
157 while (title != NULL && (temp = strchr(title, '/')) != NULL) | |
158 *temp = '-'; | |
159 | |
160 if (title == NULL || strlen(g_basename(title)) == 0) | |
161 { | |
162 g_free(title); | |
163 /* No title either. Just set it to something. */ | |
164 title = g_strdup_printf("xmms-%d", pos); | |
165 } | |
166 } | |
167 filename = g_strdup_printf("%s/%s.wav", file_path, g_basename(title)); | |
168 g_free(title); | |
169 | |
1664 | 170 output_file = vfs_fopen(filename, "wb"); |
61 | 171 g_free(filename); |
172 | |
173 if (!output_file) | |
174 return 0; | |
175 | |
176 memcpy(&header.main_chunk, "RIFF", 4); | |
177 header.length = GUINT32_TO_LE(0); | |
178 memcpy(&header.chunk_type, "WAVE", 4); | |
179 memcpy(&header.sub_chunk, "fmt ", 4); | |
180 header.sc_len = GUINT32_TO_LE(16); | |
181 header.format = GUINT16_TO_LE(1); | |
182 header.modus = GUINT16_TO_LE(nch); | |
183 header.sample_fq = GUINT32_TO_LE(rate); | |
184 if (fmt == FMT_U8 || fmt == FMT_S8) | |
185 header.bit_p_spl = GUINT16_TO_LE(8); | |
186 else | |
187 header.bit_p_spl = GUINT16_TO_LE(16); | |
188 header.byte_p_sec = GUINT32_TO_LE(rate * header.modus * (GUINT16_FROM_LE(header.bit_p_spl) / 8)); | |
189 header.byte_p_spl = GUINT16_TO_LE((GUINT16_FROM_LE(header.bit_p_spl) / (8 / nch))); | |
190 memcpy(&header.data_chunk, "data", 4); | |
191 header.data_length = GUINT32_TO_LE(0); | |
1664 | 192 vfs_fwrite(&header, sizeof (struct wavhead), 1, output_file); |
61 | 193 |
194 return 1; | |
195 } | |
196 | |
197 static void convert_buffer(gpointer buffer, gint length) | |
198 { | |
199 gint i; | |
200 | |
201 if (afmt == FMT_S8) | |
202 { | |
203 guint8 *ptr1 = buffer; | |
204 gint8 *ptr2 = buffer; | |
205 | |
206 for (i = 0; i < length; i++) | |
207 *(ptr1++) = *(ptr2++) ^ 128; | |
208 } | |
209 if (afmt == FMT_S16_BE) | |
210 { | |
211 gint16 *ptr = buffer; | |
212 | |
213 for (i = 0; i < length >> 1; i++, ptr++) | |
214 *ptr = GUINT16_SWAP_LE_BE(*ptr); | |
215 } | |
216 if (afmt == FMT_S16_NE) | |
217 { | |
218 gint16 *ptr = buffer; | |
219 | |
220 for (i = 0; i < length >> 1; i++, ptr++) | |
221 *ptr = GINT16_TO_LE(*ptr); | |
222 } | |
223 if (afmt == FMT_U16_BE) | |
224 { | |
225 gint16 *ptr1 = buffer; | |
226 guint16 *ptr2 = buffer; | |
227 | |
228 for (i = 0; i < length >> 1; i++, ptr2++) | |
229 *(ptr1++) = GINT16_TO_LE(GUINT16_FROM_BE(*ptr2) ^ 32768); | |
230 } | |
231 if (afmt == FMT_U16_LE) | |
232 { | |
233 gint16 *ptr1 = buffer; | |
234 guint16 *ptr2 = buffer; | |
235 | |
236 for (i = 0; i < length >> 1; i++, ptr2++) | |
237 *(ptr1++) = GINT16_TO_LE(GUINT16_FROM_LE(*ptr2) ^ 32768); | |
238 } | |
239 if (afmt == FMT_U16_NE) | |
240 { | |
241 gint16 *ptr1 = buffer; | |
242 guint16 *ptr2 = buffer; | |
243 | |
244 for (i = 0; i < length >> 1; i++, ptr2++) | |
245 *(ptr1++) = GINT16_TO_LE((*ptr2) ^ 32768); | |
246 } | |
247 } | |
248 | |
249 static void disk_write(void *ptr, gint length) | |
250 { | |
251 if (afmt == FMT_S8 || afmt == FMT_S16_BE || | |
252 afmt == FMT_U16_LE || afmt == FMT_U16_BE || afmt == FMT_U16_NE) | |
253 convert_buffer(ptr, length); | |
254 #ifdef WORDS_BIGENDIAN | |
255 if (afmt == FMT_S16_NE) | |
256 convert_buffer(ptr, length); | |
257 #endif | |
1664 | 258 written += vfs_fwrite(ptr, 1, length, output_file); |
61 | 259 } |
260 | |
261 static void disk_close(void) | |
262 { | |
263 if (output_file) | |
264 { | |
265 header.length = GUINT32_TO_LE(written + sizeof (struct wavhead) - 8); | |
266 | |
267 header.data_length = GUINT32_TO_LE(written); | |
1665
de0a0927a5c2
[svn] Revert broken commit r2209 and fseek -> vfs_fseek, missed in commit r2265.
chainsaw
parents:
1664
diff
changeset
|
268 vfs_fseek(output_file, 0, SEEK_SET); |
1664 | 269 vfs_fwrite(&header, sizeof (struct wavhead), 1, output_file); |
61 | 270 |
1664 | 271 vfs_fclose(output_file); |
61 | 272 written = 0; |
273 } | |
274 output_file = NULL; | |
275 } | |
276 | |
277 static void disk_flush(gint time) | |
278 { | |
279 } | |
280 | |
281 static void disk_pause(short p) | |
282 { | |
283 } | |
284 | |
285 static gint disk_free(void) | |
286 { | |
287 return 1000000; | |
288 } | |
289 | |
290 static gint disk_playing(void) | |
291 { | |
292 return 0; | |
293 } | |
294 | |
295 static gint disk_get_written_time(void) | |
296 { | |
297 if(header.byte_p_sec != 0) | |
298 return (gint) ((written * 1000) / header.byte_p_sec); | |
299 return 0; | |
300 } | |
301 | |
302 static gint disk_get_output_time(void) | |
303 { | |
304 return disk_get_written_time(); | |
305 } | |
306 | |
307 static void path_dirbrowser_cb(gchar * dir) | |
308 { | |
309 gtk_entry_set_text(GTK_ENTRY(path_entry), dir); | |
310 } | |
311 | |
312 static void path_browse_cb(GtkWidget * w, gpointer data) | |
313 { | |
314 if (!path_dirbrowser) | |
315 { | |
316 path_dirbrowser = xmms_create_dir_browser(_("Select the directory where you want to store the output files:"), file_path, GTK_SELECTION_SINGLE, path_dirbrowser_cb); | |
317 gtk_signal_connect(GTK_OBJECT(path_dirbrowser), "destroy", GTK_SIGNAL_FUNC(gtk_widget_destroyed), &path_dirbrowser); | |
318 gtk_window_set_transient_for(GTK_WINDOW(path_dirbrowser), GTK_WINDOW(configure_win)); | |
319 gtk_widget_show(path_dirbrowser); | |
320 } | |
321 } | |
322 | |
323 static void configure_ok_cb(gpointer data) | |
324 { | |
311 | 325 ConfigDb *db; |
61 | 326 |
327 if (file_path) | |
328 g_free(file_path); | |
329 file_path = g_strdup(gtk_entry_get_text(GTK_ENTRY(path_entry))); | |
330 | |
331 use_suffix = | |
332 gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(use_suffix_toggle)); | |
333 | |
311 | 334 db = bmp_cfg_db_open(); |
335 bmp_cfg_db_set_string(db, "disk_writer", "file_path", file_path); | |
336 bmp_cfg_db_set_bool(db, "disk_writer", "use_suffix", use_suffix); | |
337 bmp_cfg_db_close(db); | |
61 | 338 |
339 gtk_widget_destroy(configure_win); | |
340 if (path_dirbrowser) | |
341 gtk_widget_destroy(path_dirbrowser); | |
342 } | |
343 | |
344 static void configure_destroy(void) | |
345 { | |
346 if (path_dirbrowser) | |
347 gtk_widget_destroy(path_dirbrowser); | |
348 } | |
349 | |
350 static void disk_configure(void) | |
351 { | |
352 GtkTooltips *use_suffix_tooltips; | |
353 | |
354 if (!configure_win) | |
355 { | |
356 configure_win = gtk_window_new(GTK_WINDOW_TOPLEVEL); | |
357 | |
358 gtk_signal_connect(GTK_OBJECT(configure_win), "destroy", GTK_SIGNAL_FUNC(configure_destroy), NULL); | |
359 gtk_signal_connect(GTK_OBJECT(configure_win), "destroy", GTK_SIGNAL_FUNC(gtk_widget_destroyed), &configure_win); | |
360 gtk_window_set_title(GTK_WINDOW(configure_win), _("Disk Writer Configuration")); | |
361 gtk_window_set_position(GTK_WINDOW(configure_win), GTK_WIN_POS_MOUSE); | |
362 | |
363 gtk_container_set_border_width(GTK_CONTAINER(configure_win), 10); | |
364 | |
365 configure_vbox = gtk_vbox_new(FALSE, 10); | |
366 gtk_container_add(GTK_CONTAINER(configure_win), configure_vbox); | |
367 | |
368 path_hbox = gtk_hbox_new(FALSE, 5); | |
369 gtk_box_pack_start(GTK_BOX(configure_vbox), path_hbox, FALSE, FALSE, 0); | |
370 | |
371 path_label = gtk_label_new(_("Path:")); | |
372 gtk_box_pack_start(GTK_BOX(path_hbox), path_label, FALSE, FALSE, 0); | |
373 gtk_widget_show(path_label); | |
374 | |
375 path_entry = gtk_entry_new(); | |
376 if (file_path) | |
377 gtk_entry_set_text(GTK_ENTRY(path_entry), file_path); | |
378 gtk_widget_set_usize(path_entry, 200, -1); | |
379 gtk_box_pack_start(GTK_BOX(path_hbox), path_entry, TRUE, TRUE, 0); | |
380 gtk_widget_show(path_entry); | |
381 | |
382 path_browse = gtk_button_new_with_label(_("Browse")); | |
383 gtk_signal_connect(GTK_OBJECT(path_browse), "clicked", GTK_SIGNAL_FUNC(path_browse_cb), NULL); | |
384 gtk_box_pack_start(GTK_BOX(path_hbox), path_browse, FALSE, FALSE, 0); | |
385 gtk_widget_show(path_browse); | |
386 | |
387 gtk_widget_show(path_hbox); | |
388 | |
389 use_suffix_toggle = gtk_check_button_new_with_label(_("Don't strip file name extension")); | |
390 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(use_suffix_toggle), use_suffix); | |
391 gtk_box_pack_start(GTK_BOX(configure_vbox), use_suffix_toggle, FALSE, FALSE, 0); | |
392 use_suffix_tooltips = gtk_tooltips_new(); | |
393 gtk_tooltips_set_tip(use_suffix_tooltips, use_suffix_toggle, "If enabled, the extension from the original filename will not be stripped before adding the .wav extension to the end.", NULL); | |
394 gtk_tooltips_enable(use_suffix_tooltips); | |
395 gtk_widget_show(use_suffix_toggle); | |
396 | |
397 configure_separator = gtk_hseparator_new(); | |
398 gtk_box_pack_start(GTK_BOX(configure_vbox), configure_separator, FALSE, FALSE, 0); | |
399 gtk_widget_show(configure_separator); | |
400 | |
401 configure_bbox = gtk_hbutton_box_new(); | |
402 gtk_button_box_set_layout(GTK_BUTTON_BOX(configure_bbox), GTK_BUTTONBOX_END); | |
403 gtk_button_box_set_spacing(GTK_BUTTON_BOX(configure_bbox), 5); | |
404 gtk_box_pack_start(GTK_BOX(configure_vbox), configure_bbox, FALSE, FALSE, 0); | |
405 | |
406 configure_ok = gtk_button_new_with_label(_("OK")); | |
407 gtk_signal_connect(GTK_OBJECT(configure_ok), "clicked", GTK_SIGNAL_FUNC(configure_ok_cb), NULL); | |
408 GTK_WIDGET_SET_FLAGS(configure_ok, GTK_CAN_DEFAULT); | |
409 gtk_box_pack_start(GTK_BOX(configure_bbox), configure_ok, TRUE, TRUE, 0); | |
410 gtk_widget_show(configure_ok); | |
411 gtk_widget_grab_default(configure_ok); | |
412 | |
413 configure_cancel = gtk_button_new_with_label(_("Cancel")); | |
414 gtk_signal_connect_object(GTK_OBJECT(configure_cancel), "clicked", GTK_SIGNAL_FUNC(gtk_widget_destroy), GTK_OBJECT(configure_win)); | |
415 GTK_WIDGET_SET_FLAGS(configure_cancel, GTK_CAN_DEFAULT); | |
416 gtk_box_pack_start(GTK_BOX(configure_bbox), configure_cancel, TRUE, TRUE, 0); | |
417 gtk_widget_show(configure_cancel); | |
418 gtk_widget_show(configure_bbox); | |
419 gtk_widget_show(configure_vbox); | |
420 gtk_widget_show(configure_win); | |
421 } | |
422 } | |
543 | 423 |
424 static void disk_getvol(gint *l, gint *r) | |
425 { | |
426 (*l) = lvol; | |
427 (*r) = rvol; | |
428 } | |
429 | |
430 static void disk_setvol(gint l, gint r) | |
431 { | |
432 lvol = l; | |
433 rvol = r; | |
434 } |