Evolution consists of entirely too many small utility libraries, which increases linking and loading time, places a burden on higher layers of the application (e.g. modules) which has to remember to link to all the small in-tree utility libraries, and makes it difficult to generate API documentation for these utility libraries in one Gtk-Doc module. Merge the following utility libraries under the umbrella of libeutil, and enforce a single-include policy on libeutil so we can reorganize the files as desired without disrupting its pseudo-public API. libemail-utils/libemail-utils.la libevolution-utils/libevolution-utils.la filter/libfilter.la widgets/e-timezone-dialog/libetimezonedialog.la widgets/menus/libmenus.la widgets/misc/libemiscwidgets.la widgets/table/libetable.la widgets/text/libetext.la This also merges libedataserverui from the Evolution-Data-Server module, since Evolution is its only consumer nowadays, and I'd like to make some improvements to those APIs without concern for backward-compatibility. And finally, start a Gtk-Doc module for libeutil. It's going to be a project just getting all the symbols _listed_ much less _documented_. But the skeletal structure is in place and I'm off to a good start.
332 lines
8.4 KiB
C
332 lines
8.4 KiB
C
/*
|
|
* e-mail-formatter-audio-inline.c
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) version 3.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with the program; if not, see <http://www.gnu.org/licenses/>
|
|
*
|
|
*/
|
|
|
|
#include "e-mail-formatter-audio-inline.h"
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <glib/gi18n-lib.h>
|
|
|
|
#include <camel/camel.h>
|
|
#include <gst/gst.h>
|
|
|
|
#include <libebackend/libebackend.h>
|
|
|
|
#include <em-format/e-mail-formatter-extension.h>
|
|
#include <em-format/e-mail-formatter.h>
|
|
|
|
#include "e-mail-part-audio-inline.h"
|
|
|
|
#define d(x)
|
|
|
|
typedef EMailFormatterExtension EMailFormatterAudioInline;
|
|
typedef EMailFormatterExtensionClass EMailFormatterAudioInlineClass;
|
|
|
|
typedef EExtension EMailFormatterAudioInlineLoader;
|
|
typedef EExtensionClass EMailFormatterAudioInlineLoaderClass;
|
|
|
|
GType e_mail_formatter_audio_inline_get_type (void);
|
|
|
|
G_DEFINE_DYNAMIC_TYPE (
|
|
EMailFormatterAudioInline,
|
|
e_mail_formatter_audio_inline,
|
|
E_TYPE_MAIL_FORMATTER_EXTENSION)
|
|
|
|
static const gchar *formatter_mime_types[] = {
|
|
"application/vnd.evolution.widget.audio-inline",
|
|
"audio/ac3",
|
|
"audio/x-ac3",
|
|
"audio/basic",
|
|
"audio/mpeg",
|
|
"audio/x-mpeg",
|
|
"audio/mpeg3",
|
|
"audio/x-mpeg3",
|
|
"audio/mp3",
|
|
"audio/x-mp3",
|
|
"audio/mp4",
|
|
"audio/flac",
|
|
"audio/x-flac",
|
|
"audio/mod",
|
|
"audio/x-mod",
|
|
"audio/x-wav",
|
|
"audio/microsoft-wav",
|
|
"audio/x-wma",
|
|
"audio/x-ms-wma",
|
|
"application/ogg",
|
|
"application/x-ogg",
|
|
NULL
|
|
};
|
|
|
|
static void
|
|
pause_clicked (GtkWidget *button,
|
|
EMailPartAudioInline *part)
|
|
{
|
|
if (part->playbin) {
|
|
/* pause playing */
|
|
gst_element_set_state (part->playbin, GST_STATE_PAUSED);
|
|
}
|
|
}
|
|
|
|
static void
|
|
stop_clicked (GtkWidget *button,
|
|
EMailPartAudioInline *part)
|
|
{
|
|
if (part->playbin) {
|
|
/* ready to play */
|
|
gst_element_set_state (part->playbin, GST_STATE_READY);
|
|
part->target_state = GST_STATE_READY;
|
|
}
|
|
}
|
|
|
|
static void
|
|
set_audiosink (GstElement *playbin)
|
|
{
|
|
GstElement *audiosink;
|
|
|
|
/* now it's time to get the audio sink */
|
|
audiosink = gst_element_factory_make ("gconfaudiosink", "play_audio");
|
|
if (audiosink == NULL) {
|
|
audiosink = gst_element_factory_make ("autoaudiosink", "play_audio");
|
|
}
|
|
|
|
if (audiosink) {
|
|
g_object_set (playbin, "audio-sink", audiosink, NULL);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gst_callback (GstBus *bus,
|
|
GstMessage *message,
|
|
gpointer data)
|
|
{
|
|
EMailPartAudioInline *part = data;
|
|
GstMessageType msg_type;
|
|
|
|
g_return_val_if_fail (part != NULL, TRUE);
|
|
g_return_val_if_fail (part->playbin != NULL, TRUE);
|
|
|
|
msg_type = GST_MESSAGE_TYPE (message);
|
|
|
|
switch (msg_type) {
|
|
case GST_MESSAGE_ERROR:
|
|
gst_element_set_state (part->playbin, GST_STATE_NULL);
|
|
break;
|
|
case GST_MESSAGE_EOS:
|
|
gst_element_set_state (part->playbin, GST_STATE_READY);
|
|
break;
|
|
case GST_MESSAGE_STATE_CHANGED:
|
|
{
|
|
GstState old_state, new_state;
|
|
|
|
if (GST_MESSAGE_SRC (message) != GST_OBJECT (part->playbin))
|
|
break;
|
|
|
|
gst_message_parse_state_changed (message, &old_state, &new_state, NULL);
|
|
|
|
if (old_state == new_state)
|
|
break;
|
|
|
|
if (part->play_button)
|
|
gtk_widget_set_sensitive (
|
|
part->play_button,
|
|
new_state <= GST_STATE_PAUSED);
|
|
if (part->pause_button)
|
|
gtk_widget_set_sensitive (
|
|
part->pause_button,
|
|
new_state > GST_STATE_PAUSED);
|
|
if (part->stop_button)
|
|
gtk_widget_set_sensitive (
|
|
part->stop_button,
|
|
new_state >= GST_STATE_PAUSED);
|
|
}
|
|
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
play_clicked (GtkWidget *button,
|
|
EMailPartAudioInline *part)
|
|
{
|
|
GstState cur_state;
|
|
|
|
d (printf ("audio inline formatter: play\n"));
|
|
|
|
if (!part->filename) {
|
|
CamelStream *stream;
|
|
CamelDataWrapper *data;
|
|
GError *error = NULL;
|
|
gint argc = 1;
|
|
const gchar *argv[] = { "org_gnome_audio_inline", NULL };
|
|
|
|
/* FIXME this is ugly, we should stream this directly to gstreamer */
|
|
part->filename = e_mktemp ("org-gnome-audio-inline-file-XXXXXX");
|
|
|
|
d (printf ("audio inline formatter: write to temp file %s\n", po->filename));
|
|
|
|
stream = camel_stream_fs_new_with_name (
|
|
part->filename, O_RDWR | O_CREAT | O_TRUNC, 0600, NULL);
|
|
data = camel_medium_get_content (CAMEL_MEDIUM (part->parent.part));
|
|
camel_data_wrapper_decode_to_stream_sync (data, stream, NULL, NULL);
|
|
camel_stream_flush (stream, NULL, NULL);
|
|
g_object_unref (stream);
|
|
|
|
d (printf ("audio inline formatter: init gst playbin\n"));
|
|
|
|
if (gst_init_check (&argc, (gchar ***) &argv, &error)) {
|
|
gchar *uri;
|
|
GstBus *bus;
|
|
|
|
/* create a disk reader */
|
|
part->playbin = gst_element_factory_make ("playbin", "playbin");
|
|
if (part->playbin == NULL) {
|
|
g_printerr ("Failed to create gst_element_factory playbin; check your installation\n");
|
|
return;
|
|
|
|
}
|
|
|
|
uri = g_filename_to_uri (part->filename, NULL, NULL);
|
|
g_object_set (part->playbin, "uri", uri, NULL);
|
|
g_free (uri);
|
|
set_audiosink (part->playbin);
|
|
|
|
bus = gst_element_get_bus (part->playbin);
|
|
part->bus_id = gst_bus_add_watch (bus, gst_callback, part);
|
|
gst_object_unref (bus);
|
|
|
|
} else {
|
|
g_printerr ("GStreamer failed to initialize: %s",error ? error->message : "");
|
|
g_error_free (error);
|
|
}
|
|
}
|
|
|
|
gst_element_get_state (part->playbin, &cur_state, NULL, 0);
|
|
|
|
if (cur_state >= GST_STATE_PAUSED) {
|
|
gst_element_set_state (part->playbin, GST_STATE_READY);
|
|
}
|
|
|
|
if (part->playbin) {
|
|
/* start playing */
|
|
gst_element_set_state (part->playbin, GST_STATE_PLAYING);
|
|
}
|
|
}
|
|
|
|
static GtkWidget *
|
|
add_button (GtkWidget *box,
|
|
const gchar *stock_icon,
|
|
GCallback cb,
|
|
gpointer data,
|
|
gboolean sensitive)
|
|
{
|
|
GtkWidget *button;
|
|
|
|
button = gtk_button_new_from_stock (stock_icon);
|
|
gtk_widget_set_sensitive (button, sensitive);
|
|
g_signal_connect (button, "clicked", cb, data);
|
|
|
|
gtk_widget_show (button);
|
|
gtk_box_pack_end (GTK_BOX (box), button, TRUE, TRUE, 0);
|
|
|
|
return button;
|
|
}
|
|
|
|
static gboolean
|
|
emfe_audio_inline_format (EMailFormatterExtension *extension,
|
|
EMailFormatter *formatter,
|
|
EMailFormatterContext *context,
|
|
EMailPart *part,
|
|
CamelStream *stream,
|
|
GCancellable *cancellable)
|
|
{
|
|
gchar *str;
|
|
|
|
str = g_strdup_printf (
|
|
"<object type=\"application/vnd.evolution.widget.audio-inline\" "
|
|
"width=\"100%%\" height=\"auto\" data=\"%s\" id=\"%s\"></object>",
|
|
part->id, part->id);
|
|
|
|
camel_stream_write_string (stream, str, cancellable, NULL);
|
|
|
|
g_free (str);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GtkWidget *
|
|
emfe_audio_inline_get_widget (EMailFormatterExtension *extension,
|
|
EMailPartList *context,
|
|
EMailPart *part,
|
|
GHashTable *params)
|
|
{
|
|
GtkWidget *box;
|
|
EMailPartAudioInline *ai_part;
|
|
|
|
g_return_val_if_fail (E_MAIL_PART_IS (part, EMailPartAudioInline), NULL);
|
|
ai_part = (EMailPartAudioInline *) part;
|
|
|
|
/* it is OK to call UI functions here, since we are called from UI thread */
|
|
box = gtk_hbutton_box_new ();
|
|
ai_part->play_button = g_object_ref (
|
|
add_button (box, GTK_STOCK_MEDIA_PLAY,
|
|
G_CALLBACK (play_clicked), part, TRUE));
|
|
ai_part->pause_button = g_object_ref (
|
|
add_button (box, GTK_STOCK_MEDIA_PAUSE,
|
|
G_CALLBACK (pause_clicked), part, FALSE));
|
|
ai_part->stop_button = g_object_ref (
|
|
add_button (box, GTK_STOCK_MEDIA_STOP,
|
|
G_CALLBACK (stop_clicked), part, FALSE));
|
|
|
|
gtk_widget_show (box);
|
|
|
|
return box;
|
|
}
|
|
|
|
static void
|
|
e_mail_formatter_audio_inline_class_init (EMailFormatterExtensionClass *class)
|
|
{
|
|
class->display_name = _("Audio Player");
|
|
class->description = _("Play the attachment in embedded audio player");
|
|
class->mime_types = formatter_mime_types;
|
|
class->format = emfe_audio_inline_format;
|
|
class->get_widget = emfe_audio_inline_get_widget;
|
|
}
|
|
|
|
static void
|
|
e_mail_formatter_audio_inline_class_finalize (EMailFormatterExtensionClass *class)
|
|
{
|
|
}
|
|
|
|
static void
|
|
e_mail_formatter_audio_inline_init (EMailFormatterExtension *extension)
|
|
{
|
|
}
|
|
|
|
void
|
|
e_mail_formatter_audio_inline_type_register (GTypeModule *type_module)
|
|
{
|
|
e_mail_formatter_audio_inline_register_type (type_module);
|
|
}
|
|
|