1 /*****************************************************************************
2 * maemo_callbacks.c : Callbacks for the maemo plugin.
3 *****************************************************************************
4 * Copyright (C) 2008 the VideoLAN team
7 * Authors: Antoine Lejeune <phytos@videolan.org>
8 * Gildas Bazin <gbazin@videolan.org>
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
13 * (at your option) any later version.
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
20 * You should have received a copy of the GNU General Public License
21 * along with this program; if not, write to the Free Software
22 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
23 *****************************************************************************/
25 #include <vlc_common.h>
28 #include "maemo_callbacks.h"
30 #include <gdk/gdkkeysyms.h>
34 # include <hildon/hildon-file-chooser-dialog.h>
38 * Function used to retrieve an intf_thread_t object from a GtkWidget
40 static intf_thread_t *get_intf_from_widget( GtkWidget *widget )
42 if( GTK_IS_MENU_ITEM( widget ) )
44 /* Look for a GTK_MENU */
45 while( widget->parent && !GTK_IS_MENU( widget ) )
47 widget = widget->parent;
50 widget = gtk_menu_get_attach_widget( GTK_MENU( widget ) );
52 widget = gtk_widget_get_toplevel( GTK_WIDGET( widget ) );
53 return (intf_thread_t *)gtk_object_get_data( GTK_OBJECT( widget ),
57 gboolean delete_event_cb( GtkWidget *widget,
61 (void)event; (void)user_data;
62 intf_thread_t *p_intf = get_intf_from_widget( widget );
64 libvlc_Quit( p_intf->p_libvlc );
70 void play_cb( GtkButton *button, gpointer user_data )
73 intf_thread_t *p_intf = get_intf_from_widget( GTK_WIDGET( button ) );
75 // If there is no input, we ask the playlist to play
76 if( p_intf->p_sys->p_input == NULL )
78 playlist_Play( p_intf->p_sys->p_playlist );
82 // If there is an input, we toggle its state
84 var_Get( p_intf->p_sys->p_input, "state", &state );
85 state.i_int = ( state.i_int != PLAYING_S ) ? PLAYING_S : PAUSE_S;
86 var_Set( p_intf->p_sys->p_input, "state", state );
89 void stop_cb( GtkButton *button, gpointer user_data )
92 intf_thread_t *p_intf = get_intf_from_widget( GTK_WIDGET( button ) );
93 playlist_Stop( p_intf->p_sys->p_playlist );
96 void prev_cb( GtkButton *button, gpointer user_data )
99 intf_thread_t *p_intf = get_intf_from_widget( GTK_WIDGET( button ) );
100 playlist_Prev( p_intf->p_sys->p_playlist );
103 void next_cb( GtkButton *button, gpointer user_data )
106 intf_thread_t *p_intf = get_intf_from_widget( GTK_WIDGET( button ) );
107 playlist_Next( p_intf->p_sys->p_playlist );
110 void playlist_cb( GtkButton *button, gpointer user_data )
113 intf_thread_t *p_intf = get_intf_from_widget( GTK_WIDGET( button ) );
114 if( GTK_WIDGET_VISIBLE(p_intf->p_sys->p_playlist_window) )
116 gtk_widget_show_all( p_intf->p_sys->p_video_window );
117 gtk_widget_hide_all( p_intf->p_sys->p_playlist_window );
121 gtk_widget_hide_all( p_intf->p_sys->p_video_window );
122 gtk_widget_show_all( p_intf->p_sys->p_playlist_window );
126 void seekbar_changed_cb( GtkRange *range, GtkScrollType scroll,
127 gdouble value, gpointer data )
129 (void)scroll; (void)data;
130 intf_thread_t *p_intf = get_intf_from_widget( GTK_WIDGET( range ) );
131 if( p_intf->p_sys->p_input )
133 int i_length = hildon_seekbar_get_total_time( p_intf->p_sys->p_seekbar );
134 var_SetFloat( p_intf->p_sys->p_input, "position", (float)(value/i_length) );
138 void pl_row_activated_cb( GtkTreeView *tree_view , GtkTreePath *path,
139 GtkTreeViewColumn *column, gpointer user_data )
141 (void)column; (void)user_data;
142 intf_thread_t *p_intf = get_intf_from_widget( GTK_WIDGET( tree_view ) );
143 input_item_t *p_input;
144 GtkTreeModel *model = gtk_tree_view_get_model( tree_view );
146 gchar *filename = NULL;
148 gtk_tree_model_get_iter( model, &iter, path );
149 gtk_tree_model_get( model, &iter, 0, &filename, -1 );
151 p_input = input_item_New( p_intf, filename, NULL );
152 playlist_AddInput( p_intf->p_sys->p_playlist, p_input,
153 PLAYLIST_APPEND | PLAYLIST_GO, PLAYLIST_END, true, false );
154 vlc_gc_decref( p_input );
157 void open_cb( GtkMenuItem *menuitem, gpointer user_data )
160 intf_thread_t *p_intf = (intf_thread_t *)user_data;
161 input_item_t *p_input;
163 char *psz_filename = NULL;
165 #ifdef HAVE_HILDON_FM
166 dialog = hildon_file_chooser_dialog_new( GTK_WINDOW( p_intf->p_sys->p_main_window ),
167 GTK_FILE_CHOOSER_ACTION_OPEN );
169 dialog = gtk_file_chooser_dialog_new( "Open File", GTK_WINDOW( p_intf->p_sys->p_main_window ),
170 GTK_FILE_CHOOSER_ACTION_OPEN,
171 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
172 GTK_STOCK_OK, GTK_RESPONSE_OK, NULL );
174 gtk_widget_show_all( GTK_WIDGET( dialog ) );
176 if( gtk_dialog_run( GTK_DIALOG( dialog ) ) == GTK_RESPONSE_OK )
178 psz_filename = gtk_file_chooser_get_filename( GTK_FILE_CHOOSER( dialog ) );
182 gtk_widget_destroy( dialog );
186 gtk_widget_destroy( dialog );
188 p_input = input_item_New( p_intf, psz_filename, NULL );
189 playlist_AddInput( p_intf->p_sys->p_playlist, p_input,
190 PLAYLIST_APPEND | PLAYLIST_GO,
191 PLAYLIST_END, true, false );
192 vlc_gc_decref( p_input );
195 void open_address_cb( GtkMenuItem *menuitem, gpointer user_data )
198 intf_thread_t *p_intf = (intf_thread_t *)user_data;
199 input_item_t *p_input;
200 GtkWidget *dialog, *hbox, *label, *entry;
202 dialog = gtk_dialog_new_with_buttons( "Open Address",
203 GTK_WINDOW( p_intf->p_sys->p_main_window ),
204 GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT,
205 GTK_STOCK_OK, GTK_RESPONSE_OK,
206 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
208 label = gtk_label_new( "Address :" );
209 entry = gtk_entry_new();
210 gtk_entry_set_width_chars( GTK_ENTRY( entry ), 30 );
211 hbox = gtk_hbox_new( FALSE, 0 );
212 gtk_box_pack_start( GTK_BOX( hbox ), label, FALSE, FALSE, 0 );
213 gtk_box_pack_start( GTK_BOX( hbox ), entry, TRUE, TRUE, 0 );
214 gtk_container_add( GTK_CONTAINER( GTK_DIALOG( dialog )->vbox ), hbox );
216 gtk_widget_show_all( dialog );
217 if( gtk_dialog_run( GTK_DIALOG( dialog ) ) == GTK_RESPONSE_CANCEL )
219 gtk_widget_destroy( dialog );
223 p_input = input_item_New( p_intf,
224 gtk_entry_get_text( GTK_ENTRY( entry ) ),
226 playlist_AddInput( p_intf->p_sys->p_playlist, p_input,
227 PLAYLIST_APPEND | PLAYLIST_GO,
228 PLAYLIST_END, true, false );
229 vlc_gc_decref( p_input );
231 gtk_widget_destroy( dialog );
234 void open_webcam_cb( GtkMenuItem *menuitem, gpointer user_data )
237 intf_thread_t *p_intf = (intf_thread_t *)user_data;
238 input_item_t *p_input;
240 p_input = input_item_New( p_intf, "v4l2://", NULL );
241 playlist_AddInput( p_intf->p_sys->p_playlist, p_input,
242 PLAYLIST_APPEND | PLAYLIST_GO,
243 PLAYLIST_END, true, false );
244 vlc_gc_decref( p_input );
247 void snapshot_cb( GtkMenuItem *menuitem, gpointer user_data )
249 intf_thread_t *p_intf = (intf_thread_t *)user_data;
250 input_thread_t *p_input = p_intf->p_sys->p_input;
251 vout_thread_t *p_vout = p_input ? input_GetVout( p_input ) : NULL;
256 hildon_banner_show_information(
257 GTK_WIDGET( p_intf->p_sys->p_main_window ),
258 "gtk-dialog-error", "There is no video" );
262 var_TriggerCallback( p_vout, "video-snapshot" );
263 hildon_banner_show_information( GTK_WIDGET( p_intf->p_sys->p_main_window ),
264 NULL, "Snapshot taken" );
267 void dropframe_cb( GtkMenuItem *menuitem, gpointer user_data )
269 intf_thread_t *p_intf = (intf_thread_t *)user_data;
271 if( gtk_check_menu_item_get_active( GTK_CHECK_MENU_ITEM( menuitem ) ) )
272 config_PutInt( p_intf, "ffmpeg-skip-frame", 1 );
274 config_PutInt( p_intf, "ffmpeg-skip-frame", 0 );
277 static int keyModifiersToVLC( GdkEventKey *event )
279 int i_keyModifiers = 0;
280 if( event->state & GDK_SHIFT_MASK ) i_keyModifiers |= KEY_MODIFIER_SHIFT;
281 if( event->state & GDK_MOD1_MASK ) i_keyModifiers |= KEY_MODIFIER_ALT;
282 if( event->state & GDK_CONTROL_MASK ) i_keyModifiers |= KEY_MODIFIER_CTRL;
283 if( event->state & GDK_META_MASK ) i_keyModifiers |= KEY_MODIFIER_META;
284 return i_keyModifiers;
287 static int eventToVLCKey( GdkEventKey *event )
291 switch( event->keyval )
293 case GDK_Left: i_vlck |= KEY_LEFT; break;
294 case GDK_Right: i_vlck |= KEY_RIGHT; break;
295 case GDK_Up: i_vlck |= KEY_UP; break;
296 case GDK_Down: i_vlck |= KEY_DOWN; break;
297 case GDK_Escape: i_vlck |= KEY_ESC; break;
298 case GDK_Return: i_vlck |= KEY_ENTER; break;
300 case GDK_F1: i_vlck |= KEY_F1; break;
301 case GDK_F2: i_vlck |= KEY_F2; break;
302 case GDK_F3: i_vlck |= KEY_F3; break;
303 case GDK_F4: i_vlck |= KEY_F4; break;
304 case GDK_F5: i_vlck |= KEY_F5; break;
305 case GDK_F6: i_vlck |= KEY_F6; break;
306 case GDK_F7: i_vlck |= KEY_F7; break;
307 case GDK_F8: i_vlck |= KEY_F8; break;
308 case GDK_F9: i_vlck |= KEY_F9; break;
309 case GDK_F10: i_vlck |= KEY_F10; break;
310 case GDK_F11: i_vlck |= KEY_F11; break;
311 case GDK_F12: i_vlck |= KEY_F12; break;
313 case GDK_Page_Up: i_vlck |= KEY_PAGEUP; break;
314 case GDK_Page_Down: i_vlck |= KEY_PAGEDOWN; break;
315 case GDK_Home: i_vlck |= KEY_HOME; break;
316 case GDK_End: i_vlck |= KEY_END; break;
317 case GDK_Insert: i_vlck |= KEY_INSERT; break;
318 case GDK_Delete: i_vlck |= KEY_DELETE; break;
321 case GDK_AudioLowerVolume: i_vlck |= KEY_VOLUME_DOWN; break;
322 case GDK_AudioRaiseVolume: i_vlck |= KEY_VOLUME_UP; break;
323 case GDK_AudioMute: i_vlck |= KEY_VOLUME_MUTE; break;
324 case GDK_AudioPlay: i_vlck |= KEY_MEDIA_PLAY_PAUSE; break;
325 case GDK_AudioStop: i_vlck |= KEY_MEDIA_STOP; break;
326 case GDK_AudioNext: i_vlck |= KEY_MEDIA_NEXT_TRACK; break;
327 case GDK_AudioPrev: i_vlck |= KEY_MEDIA_PREV_TRACK; break;
333 /* Force lowercase */
334 if( event->keyval >= GDK_A && event->keyval <= GDK_Z )
335 i_vlck = event->keyval + 32;
336 /* Rest of the ascii range */
337 else if( event->keyval >= GDK_space && event->keyval <= GDK_asciitilde )
338 i_vlck = event->keyval;
341 /* Handle modifiers */
342 i_vlck |= keyModifiersToVLC( event );
347 gboolean key_cb(GtkWidget *widget, GdkEventKey *event, gpointer user_data)
349 intf_thread_t *p_intf = (intf_thread_t *)user_data;
350 widget = widget; /* unused */
352 int i_vlck = eventToVLCKey( event );
355 var_SetInteger( p_intf->p_libvlc, "key-pressed", i_vlck );
362 gboolean fullscreen_cb( gpointer user_data )
364 intf_thread_t *p_intf = (intf_thread_t *)user_data;
366 if(p_intf->p_sys->b_fullscreen)
368 gtk_widget_hide_all( GTK_WIDGET( p_intf->p_sys->p_control_window ) );
369 gtk_window_fullscreen( GTK_WINDOW(p_intf->p_sys->p_main_window) );
373 gtk_window_unfullscreen( GTK_WINDOW(p_intf->p_sys->p_main_window) );
374 gtk_widget_show_all( GTK_WIDGET( p_intf->p_sys->p_control_window ) );