]> git.sesse.net Git - vlc/blob - modules/misc/notify/growl.m
growl: fix object leak.
[vlc] / modules / misc / notify / growl.m
1 /*****************************************************************************
2  * growl.m : growl notification plugin
3  *****************************************************************************
4  * VLC specific code:
5  * 
6  * Copyright © 2008 the VideoLAN team
7  * $Id$
8  *
9  * Authors: Rafaël Carré <funman@videolanorg>
10  *
11  * This program is free software; you can redistribute it and/or modify
12  * it under the terms of the GNU General Public License as published by
13  * the Free Software Foundation; either version 2 of the License, or
14  * (at your option) any later version.
15  *
16  * This program is distributed in the hope that it will be useful,
17  * but WITHOUT ANY WARRANTY; without even the implied warranty of
18  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
19  * GNU General Public License for more details.
20  *
21  * You should have received a copy of the GNU General Public License
22  * along with this program; if not, write to the Free Software
23  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
24  *
25  * Growl specific code, ripped from growlnotify:
26  *
27  * Copyright (c) The Growl Project, 2004-2005
28  * All rights reserved.
29  *
30  * Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
31  *
32  * 1. Redistributions of source code must retain the above copyright
33  notice, this list of conditions and the following disclaimer.
34  * 2. Redistributions in binary form must reproduce the above copyright
35  notice, this list of conditions and the following disclaimer in the
36  documentation and/or other materials provided with the distribution.
37  * 3. Neither the name of Growl nor the names of its contributors
38  may be used to endorse or promote products derived from this software
39  without specific prior written permission.
40  *
41  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
42  *
43  *****************************************************************************/
44
45 /*****************************************************************************
46  * Preamble
47  *****************************************************************************/
48
49 #ifdef HAVE_CONFIG_H
50 # include "config.h"
51 #endif
52
53 #import <Foundation/Foundation.h>
54 #import <Growl/GrowlDefines.h>
55
56 #include <vlc_common.h>
57 #include <vlc_plugin.h>
58 #include <vlc_playlist.h>
59 #include <vlc_meta.h>
60 #include <vlc_interface.h>
61
62
63 /*****************************************************************************
64  * intf_sys_t
65  *****************************************************************************/
66 struct intf_sys_t
67 {
68     CFDataRef           default_icon;
69     NSAutoreleasePool   *p_pool;
70     CFStringRef         app_name;
71     CFStringRef         notification_type;
72 };
73
74 /*****************************************************************************
75  * Local prototypes
76  *****************************************************************************/
77 static int  Open    ( vlc_object_t * );
78 static void Close   ( vlc_object_t * );
79
80 static int ItemChange( vlc_object_t *, const char *,
81                        vlc_value_t, vlc_value_t, void * );
82
83 static void RegisterToGrowl( vlc_object_t * );
84 static void NotifyToGrowl( intf_thread_t *, const char *, CFDataRef );
85
86 static CFDataRef readFile(const char *);
87
88 /*****************************************************************************
89  * Module descriptor
90  ****************************************************************************/
91
92 vlc_module_begin ()
93     set_category( CAT_INTERFACE )
94     set_subcategory( SUBCAT_INTERFACE_CONTROL )
95     set_shortname( "Growl" )
96     set_description( N_("Growl Notification Plugin") )
97     set_capability( "interface", 0 )
98     set_callbacks( Open, Close )
99 vlc_module_end ()
100
101 /*****************************************************************************
102  * Open: initialize and create stuff
103  *****************************************************************************/
104 static int Open( vlc_object_t *p_this )
105 {
106     intf_thread_t *p_intf = (intf_thread_t *)p_this;
107     intf_sys_t    *p_sys;
108
109     p_sys = p_intf->p_sys = calloc( 1, sizeof(intf_sys_t) );
110     if( !p_sys )
111         return VLC_ENOMEM;
112
113     p_sys->p_pool = [[NSAutoreleasePool alloc] init];
114     p_sys->app_name = CFSTR( "VLC media player" );
115     p_sys->notification_type = CFSTR( "New input playing" );
116
117     const char *data_path = config_GetDataDir ();
118     char buf[strlen (data_path) + sizeof ("/vlc48x48.png")];
119     snprintf (buf, sizeof (buf), "%s/vlc48x48.png", data_path);
120     p_sys->default_icon = (CFDataRef) readFile( buf );
121
122     playlist_t *p_playlist = pl_Hold( p_intf );
123     var_AddCallback( p_playlist, "item-current", ItemChange, p_intf );
124     pl_Release( p_intf );
125
126     RegisterToGrowl( p_this );
127     return VLC_SUCCESS;
128 }
129
130 /*****************************************************************************
131  * Close: destroy interface stuff
132  *****************************************************************************/
133 static void Close( vlc_object_t *p_this )
134 {
135     intf_sys_t *p_sys = ((intf_thread_t*)p_this)->p_sys;
136
137     CFRelease( p_sys->default_icon );
138     CFRelease( p_sys->app_name );
139     CFRelease( p_sys->notification_type );
140     [p_sys->p_pool release];
141     free( p_sys );
142
143     playlist_t *p_playlist = pl_Hold( p_this );
144     var_DelCallback( p_playlist, "item-current", ItemChange, p_this );
145     pl_Release( p_this );
146 }
147
148 /*****************************************************************************
149  * ItemChange: Playlist item change callback
150  *****************************************************************************/
151 static int ItemChange( vlc_object_t *p_this, const char *psz_var,
152                        vlc_value_t oldval, vlc_value_t newval, void *param )
153 {
154     VLC_UNUSED(psz_var); VLC_UNUSED(oldval); VLC_UNUSED(newval);
155
156     intf_thread_t *p_intf   = (intf_thread_t*)param;
157     char *psz_tmp           = NULL;
158     char *psz_title         = NULL;
159     char *psz_artist        = NULL;
160     char *psz_album         = NULL;
161     input_thread_t *p_input;
162     p_input = playlist_CurrentInput( (playlist_t*)p_this );
163
164     if( !p_input ) return VLC_SUCCESS;
165
166     char *psz_name = input_item_GetName( input_GetItem( p_input ) );
167     if( p_input->b_dead || !psz_name )
168     {
169         /* Not playing anything ... */
170         free( psz_name );
171         vlc_object_release( p_input );
172         return VLC_SUCCESS;
173     }
174     free( psz_name );
175
176     /* Playing something ... */
177     input_item_t *p_item = input_GetItem( p_input );
178
179     psz_title = input_item_GetTitleFbName( p_item );
180     if( EMPTY_STR( psz_title ) )
181     {
182         free( psz_title );
183         vlc_object_release( p_input );
184         return VLC_SUCCESS;
185     }
186
187     psz_artist = input_item_GetArtist( p_item );
188     if( EMPTY_STR( psz_artist ) ) FREENULL( psz_artist );
189     psz_album = input_item_GetAlbum( p_item ) ;
190     if( EMPTY_STR( psz_album ) ) FREENULL( psz_album );
191
192     int i_ret;
193     if( psz_artist && psz_album )
194         i_ret = asprintf( &psz_tmp, "%s\n%s [%s]",
195                 psz_title, psz_artist, psz_album );
196     else if( psz_artist )
197         i_ret = asprintf( &psz_tmp, "%s\n%s", psz_title, psz_artist );
198     else
199         i_ret = asprintf( &psz_tmp, "%s", psz_title );
200
201     if( i_ret == -1 )
202     {
203         free( psz_title );
204         free( psz_artist );
205         free( psz_album );
206         vlc_object_release( p_input );
207         return VLC_ENOMEM;
208     }
209
210     char *psz_arturl = input_item_GetArtURL( p_item );
211     CFDataRef art = NULL;
212     if( psz_arturl && !strncmp( psz_arturl, "file://", 7 ) &&
213                     strlen( psz_arturl ) > 7 )
214         art = (CFDataRef) readFile( psz_arturl + 7 );
215
216     free( psz_title );
217     free( psz_artist );
218     free( psz_album );
219     free( psz_arturl );
220
221     NotifyToGrowl( p_intf, psz_tmp, art );
222
223     if( art ) CFRelease( art );
224
225     vlc_object_release( p_input );
226     return VLC_SUCCESS;
227 }
228
229 /*****************************************************************************
230  * RegisterToGrowl
231  *****************************************************************************/
232 static void RegisterToGrowl( vlc_object_t *p_this )
233 {
234     intf_sys_t *p_sys = ((intf_thread_t *)p_this)->p_sys;
235
236     CFArrayRef defaultAndAllNotifications = CFArrayCreate(
237         kCFAllocatorDefault, (const void **)&(p_sys->notification_type), 1,
238         &kCFTypeArrayCallBacks );
239     
240     CFTypeRef registerKeys[4] = {
241         GROWL_APP_NAME,
242         GROWL_NOTIFICATIONS_ALL,
243         GROWL_NOTIFICATIONS_DEFAULT,
244         GROWL_APP_ICON
245     };
246
247     CFTypeRef registerValues[4] = {
248         p_sys->app_name,
249         defaultAndAllNotifications,
250         defaultAndAllNotifications,
251         p_sys->default_icon
252     };
253
254     CFDictionaryRef registerInfo = CFDictionaryCreate(
255         kCFAllocatorDefault, registerKeys, registerValues, 4,
256         &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks );
257
258     CFRelease( defaultAndAllNotifications );
259
260     CFNotificationCenterPostNotificationWithOptions(
261         CFNotificationCenterGetDistributedCenter(),
262         (CFStringRef)GROWL_APP_REGISTRATION, NULL, registerInfo,
263         kCFNotificationPostToAllSessions );
264     CFRelease( registerInfo );
265 }
266
267 static void NotifyToGrowl( intf_thread_t *p_intf, const char *psz_desc, CFDataRef art )
268 {
269     intf_sys_t *p_sys = p_intf->p_sys;
270
271     CFStringRef title = CFStringCreateWithCString( kCFAllocatorDefault, _("Now playing"), kCFStringEncodingUTF8 );
272     CFStringRef desc = CFStringCreateWithCString( kCFAllocatorDefault, psz_desc, kCFStringEncodingUTF8 );
273
274     CFMutableDictionaryRef notificationInfo = CFDictionaryCreateMutable(
275         kCFAllocatorDefault, 5, &kCFTypeDictionaryKeyCallBacks,
276         &kCFTypeDictionaryValueCallBacks);
277
278     CFDictionarySetValue( notificationInfo, GROWL_NOTIFICATION_NAME, p_sys->notification_type );
279     CFDictionarySetValue( notificationInfo, GROWL_APP_NAME, p_sys->app_name );
280     CFDictionarySetValue( notificationInfo, GROWL_NOTIFICATION_TITLE, title );
281     CFDictionarySetValue( notificationInfo, GROWL_NOTIFICATION_DESCRIPTION, desc );
282
283     CFDictionarySetValue( notificationInfo, GROWL_NOTIFICATION_ICON, 
284         art ? art : p_sys->default_icon );
285
286     CFRelease( title );
287     CFRelease( desc );
288
289     CFNotificationCenterPostNotificationWithOptions(
290         CFNotificationCenterGetDistributedCenter(),
291         (CFStringRef)GROWL_NOTIFICATION, NULL, notificationInfo,
292         kCFNotificationPostToAllSessions );
293
294     CFRelease( notificationInfo );
295 }
296
297 /* Ripped from CFGrowlAdditions.c 
298  * Strangely, this function does exist in Growl shared library, but is not
299  * defined in public header files */
300 static CFDataRef readFile(const char *filename)
301 {
302     CFDataRef data;
303     // read the file into a CFDataRef
304     FILE *fp = fopen(filename, "r");
305     if( !fp )
306     return NULL;
307
308     fseek(fp, 0, SEEK_END);
309     long dataLength = ftell(fp);
310     fseek(fp, 0, SEEK_SET);
311     unsigned char *fileData = malloc(dataLength);
312     fread(fileData, 1, dataLength, fp);
313     fclose(fp);
314     return CFDataCreateWithBytesNoCopy(kCFAllocatorDefault, fileData,
315         dataLength, kCFAllocatorMalloc);
316 }
317