1 /*****************************************************************************
2 * vlcplugin.cpp: a VLC plugin for Mozilla
3 *****************************************************************************
4 * Copyright (C) 2002-2009 the VideoLAN team
7 * Authors: Samuel Hocevar <sam@zoy.org>
8 * Damien Fouilleul <damienf.fouilleul@laposte.net>
9 * Jean-Paul Saman <jpsaman@videolan.org>
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.
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.
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 *****************************************************************************/
26 /*****************************************************************************
28 *****************************************************************************/
31 #ifdef HAVE_MOZILLA_CONFIG_H
32 # include <mozilla-config.h>
35 #include "vlcplugin.h"
36 #include "control/npolibvlc.h"
40 /*****************************************************************************
41 * VlcPlugin constructor and destructor
42 *****************************************************************************/
43 VlcPlugin::VlcPlugin( NPP instance, uint16 mode ) :
50 libvlc_instance(NULL),
51 libvlc_media_list(NULL),
52 libvlc_media_player(NULL),
61 ,i_width((unsigned)-1)
62 ,i_height((unsigned)-1)
71 ,p_btnFullscreen(NULL)
76 memset(&npwindow, 0, sizeof(NPWindow));
78 memset(&npvideo, 0, sizeof(Window));
79 memset(&npcontrol, 0, sizeof(Window));
83 static bool boolValue(const char *value) {
84 return ( !strcmp(value, "1") ||
85 !strcasecmp(value, "true") ||
86 !strcasecmp(value, "yes") );
89 NPError VlcPlugin::init(int argc, char* const argn[], char* const argv[])
91 /* prepare VLC command line */
92 const char *ppsz_argv[32];
96 ppsz_argv[ppsz_argc++] = "--no-plugins-cache";
99 /* locate VLC module path */
101 ppsz_argv[ppsz_argc++] = "--plugin-path=/Library/Internet\\ Plug-Ins/VLC\\ Plugin.plugin/Contents/MacOS/modules";
102 ppsz_argv[ppsz_argc++] = "--vout=macosx";
103 #elif defined(XP_WIN)
105 DWORD i_type, i_data = MAX_PATH + 1;
106 char p_data[MAX_PATH + 1];
107 if( RegOpenKeyEx( HKEY_LOCAL_MACHINE, "Software\\VideoLAN\\VLC",
108 0, KEY_READ, &h_key ) == ERROR_SUCCESS )
110 if( RegQueryValueEx( h_key, "InstallDir", 0, &i_type,
111 (LPBYTE)p_data, &i_data ) == ERROR_SUCCESS )
113 if( i_type == REG_SZ )
115 strcat( p_data, "\\plugins" );
116 ppsz_argv[ppsz_argc++] = "--plugin-path";
117 ppsz_argv[ppsz_argc++] = p_data;
120 RegCloseKey( h_key );
122 ppsz_argv[ppsz_argc++] = "--no-one-instance";
124 #endif /* XP_MACOSX */
126 /* common settings */
127 ppsz_argv[ppsz_argc++] = "-vv";
128 ppsz_argv[ppsz_argc++] = "--no-stats";
129 ppsz_argv[ppsz_argc++] = "--no-media-library";
130 ppsz_argv[ppsz_argc++] = "--ignore-config";
131 ppsz_argv[ppsz_argc++] = "--intf=dummy";
132 ppsz_argv[ppsz_argc++] = "--no-video-title-show";
134 const char *progid = NULL;
136 /* parse plugin arguments */
137 for( int i = 0; i < argc ; i++ )
139 fprintf(stderr, "argn=%s, argv=%s\n", argn[i], argv[i]);
141 if( !strcmp( argn[i], "target" )
142 || !strcmp( argn[i], "mrl")
143 || !strcmp( argn[i], "filename")
144 || !strcmp( argn[i], "src") )
146 psz_target = argv[i];
148 else if( !strcmp( argn[i], "autoplay")
149 || !strcmp( argn[i], "autostart") )
151 b_autoplay = boolValue(argv[i]);
153 else if( !strcmp( argn[i], "fullscreen" ) )
155 if( boolValue(argv[i]) )
157 ppsz_argv[ppsz_argc++] = "--fullscreen";
161 ppsz_argv[ppsz_argc++] = "--no-fullscreen";
164 else if( !strcmp( argn[i], "mute" ) )
166 if( boolValue(argv[i]) )
168 ppsz_argv[ppsz_argc++] = "--volume=0";
171 else if( !strcmp( argn[i], "loop")
172 || !strcmp( argn[i], "autoloop") )
174 if( boolValue(argv[i]) )
176 ppsz_argv[ppsz_argc++] = "--loop";
180 ppsz_argv[ppsz_argc++] = "--no-loop";
183 else if( !strcmp( argn[i], "version")
184 || !strcmp( argn[i], "progid") )
188 else if( !strcmp( argn[i], "toolbar" ) )
190 /* FIXME: Remove this when toolbar functionality has been implemented on\
191 * MacOS X and Win32 for Firefox/Mozilla/Safari. */
193 b_toolbar = boolValue(argv[i]);
198 libvlc_exception_t ex;
199 libvlc_exception_init(&ex);
201 libvlc_instance = libvlc_new(ppsz_argc, ppsz_argv, &ex);
203 if( libvlc_exception_raised(&ex) )
205 libvlc_exception_clear(&ex);
206 return NPERR_GENERIC_ERROR;
209 libvlc_media_list = libvlc_media_list_new(libvlc_instance,&ex);
210 if( libvlc_exception_raised(&ex) )
212 libvlc_exception_clear(&ex);
213 return NPERR_GENERIC_ERROR;
217 ** fetch plugin base URL, which is the URL of the page containing the plugin
218 ** this URL is used for making absolute URL from relative URL that may be
219 ** passed as an MRL argument
223 if( NPERR_NO_ERROR == NPN_GetValue(p_browser, NPNVWindowNPObject, &plugin) )
226 ** is there a better way to get that info ?
228 static const char docLocHref[] = "document.location.href";
232 script.utf8characters = docLocHref;
233 script.utf8length = sizeof(docLocHref)-1;
235 if( NPN_Evaluate(p_browser, plugin, &script, &result) )
237 if( NPVARIANT_IS_STRING(result) )
239 NPString &location = NPVARIANT_TO_STRING(result);
241 psz_baseURL = static_cast<char*>(malloc(location.utf8length+1));
244 strncpy(psz_baseURL, location.utf8characters, location.utf8length);
245 psz_baseURL[location.utf8length] = '\0';
248 NPN_ReleaseVariantValue(&result);
250 NPN_ReleaseObject(plugin);
255 // get absolute URL from src
256 char *psz_absurl = getAbsoluteURL(psz_target);
257 psz_target = psz_absurl ? psz_absurl : strdup(psz_target);
260 /* assign plugin script root class */
262 p_scriptClass = RuntimeNPClass<LibvlcRootNPObject>::getClass();
264 return NPERR_NO_ERROR;
267 VlcPlugin::~VlcPlugin()
272 libvlc_log_close(libvlc_log, NULL);
273 if( libvlc_media_player )
274 libvlc_media_player_release( libvlc_media_player );
275 if( libvlc_media_list )
276 libvlc_media_list_release( libvlc_media_list );
277 if( libvlc_instance )
278 libvlc_release(libvlc_instance);
281 /*****************************************************************************
282 * VlcPlugin playlist replacement methods
283 *****************************************************************************/
284 void VlcPlugin::set_player_window( libvlc_exception_t *ex )
287 libvlc_media_player_set_xwindow(libvlc_media_player,
288 (libvlc_drawable_t)getVideoWindow(),
292 // XXX FIXME insert appropriate call here
295 libvlc_media_player_set_hwnd(libvlc_media_player,
301 int VlcPlugin::playlist_add( const char *mrl, libvlc_exception_t *ex )
304 libvlc_media_t *p_m = libvlc_media_new(libvlc_instance,mrl,ex);
305 if( libvlc_exception_raised(ex) )
308 libvlc_media_list_lock(libvlc_media_list);
309 libvlc_media_list_add_media(libvlc_media_list,p_m,ex);
310 if( !libvlc_exception_raised(ex) )
311 item = libvlc_media_list_count(libvlc_media_list,ex)-1;
312 libvlc_media_list_unlock(libvlc_media_list);
314 libvlc_media_release(p_m);
319 int VlcPlugin::playlist_add_extended_untrusted( const char *mrl, const char *name,
320 int optc, const char **optv, libvlc_exception_t *ex )
322 libvlc_media_t *p_m = libvlc_media_new(libvlc_instance, mrl,ex);
324 if( libvlc_exception_raised(ex) )
327 for( int i = 0; i < optc; ++i )
329 libvlc_media_add_option_untrusted(p_m, optv[i],ex);
330 if( libvlc_exception_raised(ex) )
332 libvlc_media_release(p_m);
337 libvlc_media_list_lock(libvlc_media_list);
338 libvlc_media_list_add_media(libvlc_media_list,p_m,ex);
339 if( !libvlc_exception_raised(ex) )
340 item = libvlc_media_list_count(libvlc_media_list,ex)-1;
341 libvlc_media_list_unlock(libvlc_media_list);
342 libvlc_media_release(p_m);
347 void VlcPlugin::playlist_play( libvlc_exception_t *ex )
349 if( libvlc_media_player||playlist_select(0,ex) )
350 libvlc_media_player_play(libvlc_media_player,ex);
353 void VlcPlugin::playlist_play_item( int idx, libvlc_exception_t *ex )
355 if( playlist_select(idx,ex) )
356 libvlc_media_player_play(libvlc_media_player,ex);
359 void VlcPlugin::playlist_stop( libvlc_exception_t *ex )
361 if( libvlc_media_player )
362 libvlc_media_player_stop(libvlc_media_player,ex);
365 bool VlcPlugin::playlist_select( int idx, libvlc_exception_t *ex )
367 libvlc_media_t *p_m = NULL;
369 libvlc_media_list_lock(libvlc_media_list);
371 int count = libvlc_media_list_count(libvlc_media_list,ex);
372 if( libvlc_exception_raised(ex) )
375 if( idx<0||idx>=count )
378 playlist_index = idx;
380 p_m = libvlc_media_list_item_at_index(libvlc_media_list,playlist_index,ex);
381 libvlc_media_list_unlock(libvlc_media_list);
383 if( libvlc_exception_raised(ex) )
386 if( libvlc_media_player )
388 libvlc_media_player_release( libvlc_media_player );
389 libvlc_media_player = NULL;
392 libvlc_media_player = libvlc_media_player_new_from_media(p_m,ex);
393 if( libvlc_media_player )
394 set_player_window(ex);
396 libvlc_media_release( p_m );
397 return !libvlc_exception_raised(ex);
400 libvlc_media_list_unlock(libvlc_media_list);
404 void VlcPlugin::playlist_next( libvlc_exception_t *ex )
406 if( playlist_select(playlist_index+1,ex) )
407 libvlc_media_player_play(libvlc_media_player,ex);
410 void VlcPlugin::playlist_prev( libvlc_exception_t *ex )
412 if( playlist_select(playlist_index-1,ex) )
413 libvlc_media_player_play(libvlc_media_player,ex);
416 void VlcPlugin::playlist_pause( libvlc_exception_t *ex )
418 if( libvlc_media_player )
419 libvlc_media_player_pause(libvlc_media_player,ex);
422 void VlcPlugin::playlist_delete_item( int idx, libvlc_exception_t *ex )
424 libvlc_media_list_lock(libvlc_media_list);
425 libvlc_media_list_remove_index(libvlc_media_list,idx,ex);
426 libvlc_media_list_unlock(libvlc_media_list);
429 void VlcPlugin::playlist_clear( libvlc_exception_t *ex )
431 if( libvlc_media_list )
432 libvlc_media_list_release(libvlc_media_list);
433 libvlc_media_list = libvlc_media_list_new(getVLC(),ex);
436 int VlcPlugin::playlist_count( libvlc_exception_t *ex )
439 libvlc_media_list_lock(libvlc_media_list);
440 items_count = libvlc_media_list_count(libvlc_media_list,ex);
441 libvlc_media_list_unlock(libvlc_media_list);
445 int VlcPlugin::playlist_isplaying( libvlc_exception_t *ex )
448 if( libvlc_media_player )
449 is_playing = libvlc_media_player_is_playing( libvlc_media_player, ex );
453 void VlcPlugin::toggle_fullscreen( libvlc_exception_t *ex )
455 if( playlist_isplaying(ex) )
456 libvlc_toggle_fullscreen(libvlc_media_player,ex);
459 void VlcPlugin::set_fullscreen( int yes, libvlc_exception_t *ex )
461 if( playlist_isplaying(ex) )
462 libvlc_set_fullscreen(libvlc_media_player,yes,ex);
465 int VlcPlugin::get_fullscreen( libvlc_exception_t *ex )
468 if( playlist_isplaying(ex) )
469 r = libvlc_get_fullscreen(libvlc_media_player,ex);
473 int VlcPlugin::player_has_vout( libvlc_exception_t *ex )
476 if( playlist_isplaying(ex) )
477 r = libvlc_media_player_has_vout(libvlc_media_player, ex);
481 /*****************************************************************************
483 *****************************************************************************/
485 char *VlcPlugin::getAbsoluteURL(const char *url)
489 // check whether URL is already absolute
490 const char *end=strchr(url, ':');
491 if( (NULL != end) && (end != url) )
493 // validate protocol header
494 const char *start = url;
499 while( start != end )
506 || ('/' == c)) ) /* VLC uses / to allow user to specify a demuxer */
507 // not valid protocol header, assume relative URL
511 /* we have a protocol header, therefore URL is absolute */
514 // not a valid protocol header, assume relative URL
521 size_t baseLen = strlen(psz_baseURL);
522 char *href = static_cast<char*>(malloc(baseLen+strlen(url)+1));
525 /* prepend base URL */
526 strcpy(href, psz_baseURL);
529 ** relative url could be empty,
530 ** in which case return base URL
536 ** locate pathname part of base URL
539 /* skip over protocol part */
540 char *pathstart = strchr(href, ':');
544 if( '/' == *(++pathstart) )
546 if( '/' == *(++pathstart) )
551 /* skip over host part */
552 pathstart = strchr(pathstart, '/');
553 pathend = href+baseLen;
556 // no path, add a / past end of url (over '\0')
563 /* baseURL is just a UNIX path */
566 /* baseURL is not an absolute path */
571 pathend = href+baseLen;
574 /* relative URL made of an absolute path ? */
577 /* replace path completely */
578 strcpy(pathstart, url);
582 /* find last path component and replace it */
583 while( '/' != *pathend)
587 ** if relative url path starts with one or more '../',
588 ** factor them out of href so that we return a
591 while( pathend != pathstart )
599 /* relative url is just '.' */
605 /* relative url starts with './' */
614 /* relative url is '..' */
620 /* relative url starts with '../' */
628 while( '/' != *pathend );
630 /* skip over '/' separator */
632 /* concatenate remaining base URL and relative URL */
633 strcpy(pathend, url);
642 int VlcPlugin::setSize(unsigned width, unsigned height)
644 int diff = (width != i_width) || (height != i_height);
653 #define BTN_SPACE ((unsigned int)4)
654 void VlcPlugin::showToolbar()
656 const NPWindow& window = getWindow();
657 Window control = getControlWindow();
658 Window video = getVideoWindow();
659 Display *p_display = ((NPSetWindowCallbackStruct *)window.ws_info)->display;
660 unsigned int i_height = 0, i_width = BTN_SPACE;
664 XpmReadFileToImage( p_display, DATA_PATH "/mozilla/play.xpm",
665 &p_btnPlay, NULL, NULL);
668 i_height = __MAX( i_height, p_btnPlay->height );
671 XpmReadFileToImage( p_display, DATA_PATH "/mozilla/pause.xpm",
672 &p_btnPause, NULL, NULL);
675 i_height = __MAX( i_height, p_btnPause->height );
677 i_width += __MAX( p_btnPause->width, p_btnPlay->width );
680 XpmReadFileToImage( p_display, DATA_PATH "/mozilla/stop.xpm",
681 &p_btnStop, NULL, NULL );
684 i_height = __MAX( i_height, p_btnStop->height );
685 i_width += BTN_SPACE + p_btnStop->width;
688 XpmReadFileToImage( p_display, DATA_PATH "/mozilla/time_line.xpm",
689 &p_timeline, NULL, NULL);
692 i_height = __MAX( i_height, p_timeline->height );
693 i_width += BTN_SPACE + p_timeline->width;
696 XpmReadFileToImage( p_display, DATA_PATH "/mozilla/time_icon.xpm",
697 &p_btnTime, NULL, NULL);
700 i_height = __MAX( i_height, p_btnTime->height );
701 i_width += BTN_SPACE + p_btnTime->width;
703 if( !p_btnFullscreen )
704 XpmReadFileToImage( p_display, DATA_PATH "/mozilla/fullscreen.xpm",
705 &p_btnFullscreen, NULL, NULL);
706 if( p_btnFullscreen )
708 i_height = __MAX( i_height, p_btnFullscreen->height );
709 i_width += BTN_SPACE + p_btnFullscreen->width;
712 XpmReadFileToImage( p_display, DATA_PATH "/mozilla/volume_max.xpm",
713 &p_btnMute, NULL, NULL);
716 i_height = __MAX( i_height, p_btnMute->height );
719 XpmReadFileToImage( p_display, DATA_PATH "/mozilla/volume_mute.xpm",
720 &p_btnUnmute, NULL, NULL);
723 i_height = __MAX( i_height, p_btnUnmute->height );
725 i_width += BTN_SPACE + __MAX( p_btnUnmute->width, p_btnMute->width );
727 setToolbarSize( i_width, i_height );
729 if( !p_btnPlay || !p_btnPause || !p_btnStop || !p_timeline ||
730 !p_btnTime || !p_btnFullscreen || !p_btnMute || !p_btnUnmute )
731 fprintf(stderr, "Error: some button images not found in %s\n", DATA_PATH );
733 /* reset panels position and size */
734 /* XXX use i_width */
735 XResizeWindow( p_display, video, window.width, window.height - i_height);
736 XMoveWindow( p_display, control, 0, window.height - i_height );
737 XResizeWindow( p_display, control, window.width, i_height -1);
739 b_toolbar = 1; /* says toolbar is now shown */
743 void VlcPlugin::hideToolbar()
745 const NPWindow& window = getWindow();
746 Display *p_display = ((NPSetWindowCallbackStruct *)window.ws_info)->display;
747 Window control = getControlWindow();
748 Window video = getVideoWindow();
750 i_tb_width = i_tb_height = 0;
752 if( p_btnPlay ) XDestroyImage( p_btnPlay );
753 if( p_btnPause ) XDestroyImage( p_btnPause );
754 if( p_btnStop ) XDestroyImage( p_btnStop );
755 if( p_timeline ) XDestroyImage( p_timeline );
756 if( p_btnTime ) XDestroyImage( p_btnTime );
757 if( p_btnFullscreen ) XDestroyImage( p_btnFullscreen );
758 if( p_btnMute ) XDestroyImage( p_btnMute );
759 if( p_btnUnmute ) XDestroyImage( p_btnUnmute );
766 p_btnFullscreen = NULL;
770 /* reset panels position and size */
771 /* XXX use i_width */
772 XResizeWindow( p_display, video, window.width, window.height );
773 XMoveWindow( p_display, control, 0, window.height-1 );
774 XResizeWindow( p_display, control, window.width, 1 );
776 b_toolbar = 0; /* says toolbar is now hidden */
780 void VlcPlugin::redrawToolbar()
782 libvlc_exception_t ex;
785 unsigned int dst_x, dst_y;
788 unsigned int i_tb_width, i_tb_height;
790 /* This method does nothing if toolbar is hidden. */
794 const NPWindow& window = getWindow();
795 Window control = getControlWindow();
796 Display *p_display = ((NPSetWindowCallbackStruct *)window.ws_info)->display;
798 getToolbarSize( &i_tb_width, &i_tb_height );
800 libvlc_exception_init( &ex );
803 b_mute = libvlc_audio_get_mute( getVLC(), &ex );
804 libvlc_exception_clear( &ex );
806 gcv.foreground = BlackPixel( p_display, 0 );
807 gc = XCreateGC( p_display, control, GCForeground, &gcv );
809 XFillRectangle( p_display, control, gc,
810 0, 0, window.width, i_tb_height );
811 gcv.foreground = WhitePixel( p_display, 0 );
812 XChangeGC( p_display, gc, GCForeground, &gcv );
816 dst_y = i_tb_height >> 1; /* baseline = vertical middle */
818 if( p_btnPause && (is_playing == 1) )
820 XPutImage( p_display, control, gc, p_btnPause, 0, 0, dst_x,
821 dst_y - (p_btnPause->height >> 1),
822 p_btnPause->width, p_btnPause->height );
823 dst_x += BTN_SPACE + p_btnPause->width;
827 XPutImage( p_display, control, gc, p_btnPlay, 0, 0, dst_x,
828 dst_y - (p_btnPlay->height >> 1),
829 p_btnPlay->width, p_btnPlay->height );
830 dst_x += BTN_SPACE + p_btnPlay->width;
834 XPutImage( p_display, control, gc, p_btnStop, 0, 0, dst_x,
835 dst_y - (p_btnStop->height >> 1),
836 p_btnStop->width, p_btnStop->height );
838 dst_x += BTN_SPACE + ( p_btnStop ? p_btnStop->width : 0 );
840 if( p_btnFullscreen )
841 XPutImage( p_display, control, gc, p_btnFullscreen, 0, 0, dst_x,
842 dst_y - (p_btnFullscreen->height >> 1),
843 p_btnFullscreen->width, p_btnFullscreen->height );
845 dst_x += BTN_SPACE + ( p_btnFullscreen ? p_btnFullscreen->width : 0 );
847 if( p_btnUnmute && b_mute )
849 XPutImage( p_display, control, gc, p_btnUnmute, 0, 0, dst_x,
850 dst_y - (p_btnUnmute->height >> 1),
851 p_btnUnmute->width, p_btnUnmute->height );
853 dst_x += BTN_SPACE + ( p_btnUnmute ? p_btnUnmute->width : 0 );
857 XPutImage( p_display, control, gc, p_btnMute, 0, 0, dst_x,
858 dst_y - (p_btnMute->height >> 1),
859 p_btnMute->width, p_btnMute->height );
861 dst_x += BTN_SPACE + ( p_btnMute ? p_btnMute->width : 0 );
865 XPutImage( p_display, control, gc, p_timeline, 0, 0, dst_x,
866 dst_y - (p_timeline->height >> 1),
867 (window.width-(dst_x+BTN_SPACE)), p_timeline->height );
869 /* get movie position in % */
870 if( playlist_isplaying(&ex) )
872 i_last_position = (int)((window.width-(dst_x+BTN_SPACE))*
873 libvlc_media_player_get_position(libvlc_media_player,&ex));
875 libvlc_exception_clear( &ex );
878 XPutImage( p_display, control, gc, p_btnTime,
879 0, 0, (dst_x+i_last_position),
880 dst_y - (p_btnTime->height >> 1),
881 p_btnTime->width, p_btnTime->height );
883 XFreeGC( p_display, gc );
886 vlc_toolbar_clicked_t VlcPlugin::getToolbarButtonClicked( int i_xpos, int i_ypos )
888 unsigned int i_dest = BTN_SPACE;
891 libvlc_exception_t ex;
894 fprintf( stderr, "ToolbarButtonClicked:: "
895 "trying to match (%d,%d) (%d,%d)\n",
896 i_xpos, i_ypos, i_tb_height, i_tb_width );
898 if( i_ypos >= i_tb_width )
899 return clicked_Unknown;
901 /* Note: the order of testing is dependend on the original
902 * drawing positions of the icon buttons. Buttons are tested
907 libvlc_exception_init( &ex );
908 is_playing = playlist_isplaying( &ex );
909 libvlc_exception_clear( &ex );
912 b_mute = libvlc_audio_get_mute( getVLC(), &ex );
913 libvlc_exception_clear( &ex );
915 /* is Pause of Play button clicked */
916 if( (is_playing != 1) &&
917 (i_xpos >= (BTN_SPACE>>1)) &&
918 (i_xpos <= i_dest + p_btnPlay->width + (BTN_SPACE>>1)) )
920 else if( (i_xpos >= (BTN_SPACE>>1)) &&
921 (i_xpos <= i_dest + p_btnPause->width) )
922 return clicked_Pause;
924 /* is Stop button clicked */
925 if( is_playing != 1 )
926 i_dest += (p_btnPlay->width + (BTN_SPACE>>1));
928 i_dest += (p_btnPause->width + (BTN_SPACE>>1));
930 if( (i_xpos >= i_dest) &&
931 (i_xpos <= i_dest + p_btnStop->width + (BTN_SPACE>>1)) )
934 /* is Fullscreen button clicked */
935 i_dest += (p_btnStop->width + (BTN_SPACE>>1));
936 if( (i_xpos >= i_dest) &&
937 (i_xpos <= i_dest + p_btnFullscreen->width + (BTN_SPACE>>1)) )
938 return clicked_Fullscreen;
940 /* is Mute or Unmute button clicked */
941 i_dest += (p_btnFullscreen->width + (BTN_SPACE>>1));
942 if( !b_mute && (i_xpos >= i_dest) &&
943 (i_xpos <= i_dest + p_btnMute->width + (BTN_SPACE>>1)) )
945 else if( (i_xpos >= i_dest) &&
946 (i_xpos <= i_dest + p_btnUnmute->width + (BTN_SPACE>>1)) )
947 return clicked_Unmute;
949 /* is timeline clicked */
951 i_dest += (p_btnMute->width + (BTN_SPACE>>1));
953 i_dest += (p_btnUnmute->width + (BTN_SPACE>>1));
954 if( (i_xpos >= i_dest) &&
955 (i_xpos <= i_dest + p_timeline->width + (BTN_SPACE>>1)) )
956 return clicked_timeline;
958 /* is time button clicked */
959 i_dest += (p_timeline->width + (BTN_SPACE>>1));
960 if( (i_xpos >= i_dest) &&
961 (i_xpos <= i_dest + p_btnTime->width + (BTN_SPACE>>1)) )
964 return clicked_Unknown;