1 /*****************************************************************************
2 * npolibvlc.cpp: official Javascript APIs
3 *****************************************************************************
4 * Copyright (C) 2002-2009 the VideoLAN team
6 * Authors: Damien Fouilleul <Damien.Fouilleul@laposte.net>
7 * JP Dinger <jpd@m2x.nl>
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.
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.
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
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22 *****************************************************************************/
31 #ifdef HAVE_MOZILLA_CONFIG_H
32 # include <mozilla-config.h>
35 #include "vlcplugin.h"
36 #include "npolibvlc.h"
39 ** Local helper macros and function
41 #define COUNTNAMES(a,b,c) const int a::b = sizeof(a::c)/sizeof(NPUTF8 *)
42 #define RETURN_ON_EXCEPTION(this,ex) \
43 do { if( libvlc_exception_raised(&ex) ) \
45 NPN_SetException(this, libvlc_exception_get_message(&ex)); \
46 libvlc_exception_clear(&ex); \
47 return INVOKERESULT_GENERIC_ERROR; \
51 ** implementation of libvlc root object
54 LibvlcRootNPObject::~LibvlcRootNPObject()
57 ** When the plugin is destroyed, firefox takes it upon itself to
58 ** destroy all 'live' script objects and ignores refcounting.
59 ** Therefore we cannot safely assume that refcounting will control
60 ** lifespan of objects. Hence they are only lazily created on
61 ** request, so that firefox can take ownership, and are not released
62 ** when the plugin is destroyed.
66 if( audioObj ) NPN_ReleaseObject(audioObj);
67 if( inputObj ) NPN_ReleaseObject(inputObj);
68 if( playlistObj ) NPN_ReleaseObject(playlistObj);
69 if( videoObj ) NPN_ReleaseObject(videoObj);
73 const NPUTF8 * const LibvlcRootNPObject::propertyNames[] =
82 COUNTNAMES(LibvlcRootNPObject,propertyCount,propertyNames);
84 enum LibvlcRootNPObjectPropertyIds
94 RuntimeNPObject::InvokeResult
95 LibvlcRootNPObject::getProperty(int index, NPVariant &result)
97 /* is plugin still running */
98 if( isPluginRunning() )
103 // create child object in lazyman fashion to avoid
104 // ownership problem with firefox
106 audioObj = NPN_CreateObject(_instance,
107 RuntimeNPClass<LibvlcAudioNPObject>::getClass());
108 OBJECT_TO_NPVARIANT(NPN_RetainObject(audioObj), result);
109 return INVOKERESULT_NO_ERROR;
111 // create child object in lazyman fashion to avoid
112 // ownership problem with firefox
114 inputObj = NPN_CreateObject(_instance,
115 RuntimeNPClass<LibvlcInputNPObject>::getClass());
116 OBJECT_TO_NPVARIANT(NPN_RetainObject(inputObj), result);
117 return INVOKERESULT_NO_ERROR;
118 case ID_root_playlist:
119 // create child object in lazyman fashion to avoid
120 // ownership problem with firefox
122 playlistObj = NPN_CreateObject(_instance,
123 RuntimeNPClass<LibvlcPlaylistNPObject>::getClass());
124 OBJECT_TO_NPVARIANT(NPN_RetainObject(playlistObj), result);
125 return INVOKERESULT_NO_ERROR;
126 case ID_root_subtitle:
127 // create child object in lazyman fashion to avoid
128 // ownership problem with firefox
130 subtitleObj = NPN_CreateObject(_instance,
131 RuntimeNPClass<LibvlcSubtitleNPObject>::getClass());
132 OBJECT_TO_NPVARIANT(NPN_RetainObject(subtitleObj), result);
133 return INVOKERESULT_NO_ERROR;
135 // create child object in lazyman fashion to avoid
136 // ownership problem with firefox
138 videoObj = NPN_CreateObject(_instance,
139 RuntimeNPClass<LibvlcVideoNPObject>::getClass());
140 OBJECT_TO_NPVARIANT(NPN_RetainObject(videoObj), result);
141 return INVOKERESULT_NO_ERROR;
142 case ID_root_VersionInfo:
143 return invokeResultString(libvlc_get_version(),result);
148 return INVOKERESULT_GENERIC_ERROR;
151 const NPUTF8 * const LibvlcRootNPObject::methodNames[] =
155 COUNTNAMES(LibvlcRootNPObject,methodCount,methodNames);
157 enum LibvlcRootNPObjectMethodIds
162 RuntimeNPObject::InvokeResult LibvlcRootNPObject::invoke(int index,
163 const NPVariant *args, uint32_t argCount, NPVariant &result)
165 /* is plugin still running */
166 if( isPluginRunning() )
168 libvlc_exception_t ex;
169 libvlc_exception_init(&ex);
173 case ID_root_versionInfo:
175 return INVOKERESULT_NO_SUCH_METHOD;
176 return invokeResultString(libvlc_get_version(),result);
181 return INVOKERESULT_GENERIC_ERROR;
185 ** implementation of libvlc audio object
188 const NPUTF8 * const LibvlcAudioNPObject::propertyNames[] =
196 COUNTNAMES(LibvlcAudioNPObject,propertyCount,propertyNames);
198 enum LibvlcAudioNPObjectPropertyIds
207 RuntimeNPObject::InvokeResult
208 LibvlcAudioNPObject::getProperty(int index, NPVariant &result)
210 /* is plugin still running */
211 if( isPluginRunning() )
213 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
214 libvlc_exception_t ex;
215 libvlc_exception_init(&ex);
221 bool muted = libvlc_audio_get_mute(p_plugin->getVLC(), &ex);
222 RETURN_ON_EXCEPTION(this,ex);
223 BOOLEAN_TO_NPVARIANT(muted, result);
224 return INVOKERESULT_NO_ERROR;
226 case ID_audio_volume:
228 int volume = libvlc_audio_get_volume(p_plugin->getVLC(), &ex);
229 RETURN_ON_EXCEPTION(this,ex);
230 INT32_TO_NPVARIANT(volume, result);
231 return INVOKERESULT_NO_ERROR;
235 libvlc_media_player_t *p_md = p_plugin->getMD(&ex);
236 RETURN_ON_EXCEPTION(this,ex);
237 int track = libvlc_audio_get_track(p_md, &ex);
238 RETURN_ON_EXCEPTION(this,ex);
239 INT32_TO_NPVARIANT(track, result);
240 return INVOKERESULT_NO_ERROR;
244 libvlc_media_player_t *p_md = p_plugin->getMD(&ex);
245 RETURN_ON_EXCEPTION(this,ex);
246 // get the number of audio track available
247 int i_track = libvlc_audio_get_track_count(p_md, &ex);
248 RETURN_ON_EXCEPTION(this,ex);
250 INT32_TO_NPVARIANT(i_track, result);
251 return INVOKERESULT_NO_ERROR;
253 case ID_audio_channel:
255 int channel = libvlc_audio_get_channel(p_plugin->getVLC(), &ex);
256 RETURN_ON_EXCEPTION(this,ex);
257 INT32_TO_NPVARIANT(channel, result);
258 return INVOKERESULT_NO_ERROR;
264 return INVOKERESULT_GENERIC_ERROR;
267 RuntimeNPObject::InvokeResult
268 LibvlcAudioNPObject::setProperty(int index, const NPVariant &value)
270 /* is plugin still running */
271 if( isPluginRunning() )
273 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
274 libvlc_exception_t ex;
275 libvlc_exception_init(&ex);
280 if( NPVARIANT_IS_BOOLEAN(value) )
282 libvlc_audio_set_mute(p_plugin->getVLC(),
283 NPVARIANT_TO_BOOLEAN(value), &ex);
284 RETURN_ON_EXCEPTION(this,ex);
285 return INVOKERESULT_NO_ERROR;
287 return INVOKERESULT_INVALID_VALUE;
288 case ID_audio_volume:
289 if( isNumberValue(value) )
291 libvlc_audio_set_volume(p_plugin->getVLC(),
292 numberValue(value), &ex);
293 RETURN_ON_EXCEPTION(this,ex);
294 return INVOKERESULT_NO_ERROR;
296 return INVOKERESULT_INVALID_VALUE;
298 if( isNumberValue(value) )
300 libvlc_media_player_t *p_md = p_plugin->getMD(&ex);
301 RETURN_ON_EXCEPTION(this,ex);
302 libvlc_audio_set_track(p_md, numberValue(value), &ex);
303 RETURN_ON_EXCEPTION(this,ex);
304 return INVOKERESULT_NO_ERROR;
306 return INVOKERESULT_INVALID_VALUE;
307 case ID_audio_channel:
308 if( isNumberValue(value) )
310 libvlc_audio_set_channel(p_plugin->getVLC(),
311 numberValue(value), &ex);
312 RETURN_ON_EXCEPTION(this,ex);
313 return INVOKERESULT_NO_ERROR;
315 return INVOKERESULT_INVALID_VALUE;
320 return INVOKERESULT_GENERIC_ERROR;
323 const NPUTF8 * const LibvlcAudioNPObject::methodNames[] =
328 COUNTNAMES(LibvlcAudioNPObject,methodCount,methodNames);
330 enum LibvlcAudioNPObjectMethodIds
333 ID_audio_description,
336 RuntimeNPObject::InvokeResult
337 LibvlcAudioNPObject::invoke(int index, const NPVariant *args,
338 uint32_t argCount, NPVariant &result)
340 /* is plugin still running */
341 if( isPluginRunning() )
343 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
344 libvlc_exception_t ex;
345 libvlc_exception_init(&ex);
349 case ID_audio_togglemute:
352 libvlc_audio_toggle_mute(p_plugin->getVLC(), &ex);
353 RETURN_ON_EXCEPTION(this,ex);
354 VOID_TO_NPVARIANT(result);
355 return INVOKERESULT_NO_ERROR;
357 return INVOKERESULT_NO_SUCH_METHOD;
358 case ID_audio_description:
363 int i_trackID, i_limit, i;
364 libvlc_track_description_t *p_trackDesc;
366 libvlc_media_player_t *p_md = p_plugin->getMD(&ex);
367 RETURN_ON_EXCEPTION(this,ex);
369 /* get tracks description */
370 p_trackDesc = libvlc_audio_get_track_description(p_md, &ex);
371 RETURN_ON_EXCEPTION(this,ex);
373 return INVOKERESULT_GENERIC_ERROR;
375 /* get the number of track available */
376 i_limit = libvlc_audio_get_track_count(p_md, &ex);
377 RETURN_ON_EXCEPTION(this,ex);
379 /* check if a number is given by the user
380 * and get the track number */
381 if( isNumberValue(args[0]) )
382 i_trackID = numberValue(args[0]);
384 return INVOKERESULT_INVALID_VALUE;
386 /* if bad number is given return invalid value */
387 if ( ( i_trackID > ( i_limit - 1 ) ) || ( i_trackID < 0 ) )
388 return INVOKERESULT_INVALID_VALUE;
390 /* get the good trackDesc */
391 for( i = 0 ; i < i_trackID ; i++ )
393 p_trackDesc = p_trackDesc->p_next;
395 psz_name = p_trackDesc->psz_name;
397 /* display the name of the track chosen */
398 return invokeResultString( psz_name, result );
400 return INVOKERESULT_NO_SUCH_METHOD;
406 return INVOKERESULT_GENERIC_ERROR;
410 ** implementation of libvlc input object
413 const NPUTF8 * const LibvlcInputNPObject::propertyNames[] =
423 COUNTNAMES(LibvlcInputNPObject,propertyCount,propertyNames);
425 enum LibvlcInputNPObjectPropertyIds
436 RuntimeNPObject::InvokeResult
437 LibvlcInputNPObject::getProperty(int index, NPVariant &result)
439 /* is plugin still running */
440 if( isPluginRunning() )
442 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
443 libvlc_exception_t ex;
444 libvlc_exception_init(&ex);
446 libvlc_media_player_t *p_md = p_plugin->getMD(&ex);
447 if( libvlc_exception_raised(&ex) )
449 if( index != ID_input_state )
451 NPN_SetException(this, libvlc_exception_get_message(&ex));
452 libvlc_exception_clear(&ex);
453 return INVOKERESULT_GENERIC_ERROR;
457 /* for input state, return CLOSED rather than an exception */
458 INT32_TO_NPVARIANT(0, result);
459 libvlc_exception_clear(&ex);
460 return INVOKERESULT_NO_ERROR;
466 case ID_input_length:
468 double val = (double)libvlc_media_player_get_length(p_md, &ex);
469 RETURN_ON_EXCEPTION(this,ex);
470 DOUBLE_TO_NPVARIANT(val, result);
471 return INVOKERESULT_NO_ERROR;
473 case ID_input_position:
475 double val = libvlc_media_player_get_position(p_md, &ex);
476 RETURN_ON_EXCEPTION(this,ex);
477 DOUBLE_TO_NPVARIANT(val, result);
478 return INVOKERESULT_NO_ERROR;
482 double val = (double)libvlc_media_player_get_time(p_md, &ex);
483 RETURN_ON_EXCEPTION(this,ex);
484 DOUBLE_TO_NPVARIANT(val, result);
485 return INVOKERESULT_NO_ERROR;
489 int val = libvlc_media_player_get_state(p_md, &ex);
490 RETURN_ON_EXCEPTION(this,ex);
491 INT32_TO_NPVARIANT(val, result);
492 return INVOKERESULT_NO_ERROR;
496 float val = libvlc_media_player_get_rate(p_md, &ex);
497 RETURN_ON_EXCEPTION(this,ex);
498 DOUBLE_TO_NPVARIANT(val, result);
499 return INVOKERESULT_NO_ERROR;
503 double val = libvlc_media_player_get_fps(p_md, &ex);
504 RETURN_ON_EXCEPTION(this,ex);
505 DOUBLE_TO_NPVARIANT(val, result);
506 return INVOKERESULT_NO_ERROR;
508 case ID_input_hasvout:
510 bool val = p_plugin->player_has_vout(&ex);
511 RETURN_ON_EXCEPTION(this,ex);
512 BOOLEAN_TO_NPVARIANT(val, result);
513 return INVOKERESULT_NO_ERROR;
519 return INVOKERESULT_GENERIC_ERROR;
522 RuntimeNPObject::InvokeResult
523 LibvlcInputNPObject::setProperty(int index, const NPVariant &value)
525 /* is plugin still running */
526 if( isPluginRunning() )
528 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
529 libvlc_exception_t ex;
530 libvlc_exception_init(&ex);
532 libvlc_media_player_t *p_md = p_plugin->getMD(&ex);
533 RETURN_ON_EXCEPTION(this,ex);
537 case ID_input_position:
539 if( ! NPVARIANT_IS_DOUBLE(value) )
541 return INVOKERESULT_INVALID_VALUE;
544 float val = (float)NPVARIANT_TO_DOUBLE(value);
545 libvlc_media_player_set_position(p_md, val, &ex);
546 RETURN_ON_EXCEPTION(this,ex);
547 return INVOKERESULT_NO_ERROR;
552 if( NPVARIANT_IS_INT32(value) )
553 val = (int64_t)NPVARIANT_TO_INT32(value);
554 else if( NPVARIANT_IS_DOUBLE(value) )
555 val = (int64_t)NPVARIANT_TO_DOUBLE(value);
558 return INVOKERESULT_INVALID_VALUE;
561 libvlc_media_player_set_time(p_md, val, &ex);
562 RETURN_ON_EXCEPTION(this,ex);
563 return INVOKERESULT_NO_ERROR;
568 if( NPVARIANT_IS_INT32(value) )
569 val = (float)NPVARIANT_TO_INT32(value);
570 else if( NPVARIANT_IS_DOUBLE(value) )
571 val = (float)NPVARIANT_TO_DOUBLE(value);
574 return INVOKERESULT_INVALID_VALUE;
577 libvlc_media_player_set_rate(p_md, val, &ex);
578 RETURN_ON_EXCEPTION(this,ex);
579 return INVOKERESULT_NO_ERROR;
585 return INVOKERESULT_GENERIC_ERROR;
588 const NPUTF8 * const LibvlcInputNPObject::methodNames[] =
593 COUNTNAMES(LibvlcInputNPObject,methodCount,methodNames);
596 ** implementation of libvlc playlist items object
599 const NPUTF8 * const LibvlcPlaylistItemsNPObject::propertyNames[] =
603 COUNTNAMES(LibvlcPlaylistItemsNPObject,propertyCount,propertyNames);
605 enum LibvlcPlaylistItemsNPObjectPropertyIds
607 ID_playlistitems_count,
610 RuntimeNPObject::InvokeResult
611 LibvlcPlaylistItemsNPObject::getProperty(int index, NPVariant &result)
613 /* is plugin still running */
614 if( isPluginRunning() )
616 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
617 libvlc_exception_t ex;
618 libvlc_exception_init(&ex);
622 case ID_playlistitems_count:
624 int val = p_plugin->playlist_count(&ex);
625 RETURN_ON_EXCEPTION(this,ex);
626 INT32_TO_NPVARIANT(val, result);
627 return INVOKERESULT_NO_ERROR;
633 return INVOKERESULT_GENERIC_ERROR;
636 const NPUTF8 * const LibvlcPlaylistItemsNPObject::methodNames[] =
641 COUNTNAMES(LibvlcPlaylistItemsNPObject,methodCount,methodNames);
643 enum LibvlcPlaylistItemsNPObjectMethodIds
645 ID_playlistitems_clear,
646 ID_playlistitems_remove,
649 RuntimeNPObject::InvokeResult
650 LibvlcPlaylistItemsNPObject::invoke(int index, const NPVariant *args,
651 uint32_t argCount, NPVariant &result)
653 /* is plugin still running */
654 if( isPluginRunning() )
656 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
657 libvlc_exception_t ex;
658 libvlc_exception_init(&ex);
662 case ID_playlistitems_clear:
665 p_plugin->playlist_clear(&ex);
666 RETURN_ON_EXCEPTION(this,ex);
667 VOID_TO_NPVARIANT(result);
668 return INVOKERESULT_NO_ERROR;
670 return INVOKERESULT_NO_SUCH_METHOD;
671 case ID_playlistitems_remove:
672 if( (argCount == 1) && isNumberValue(args[0]) )
674 p_plugin->playlist_delete_item(numberValue(args[0]),&ex);
675 RETURN_ON_EXCEPTION(this,ex);
676 VOID_TO_NPVARIANT(result);
677 return INVOKERESULT_NO_ERROR;
679 return INVOKERESULT_NO_SUCH_METHOD;
684 return INVOKERESULT_GENERIC_ERROR;
688 ** implementation of libvlc playlist object
691 LibvlcPlaylistNPObject::~LibvlcPlaylistNPObject()
693 // Why the isValid()?
694 if( isValid() && playlistItemsObj )
695 NPN_ReleaseObject(playlistItemsObj);
698 const NPUTF8 * const LibvlcPlaylistNPObject::propertyNames[] =
700 "itemCount", /* deprecated */
704 COUNTNAMES(LibvlcPlaylistNPObject,propertyCount,propertyNames);
706 enum LibvlcPlaylistNPObjectPropertyIds
708 ID_playlist_itemcount,
709 ID_playlist_isplaying,
713 RuntimeNPObject::InvokeResult
714 LibvlcPlaylistNPObject::getProperty(int index, NPVariant &result)
716 /* is plugin still running */
717 if( isPluginRunning() )
719 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
720 libvlc_exception_t ex;
721 libvlc_exception_init(&ex);
725 case ID_playlist_itemcount: /* deprecated */
727 int val = p_plugin->playlist_count(&ex);
728 RETURN_ON_EXCEPTION(this,ex);
729 INT32_TO_NPVARIANT(val, result);
730 return INVOKERESULT_NO_ERROR;
732 case ID_playlist_isplaying:
734 int val = p_plugin->playlist_isplaying(&ex);
735 RETURN_ON_EXCEPTION(this,ex);
736 BOOLEAN_TO_NPVARIANT(val, result);
737 return INVOKERESULT_NO_ERROR;
739 case ID_playlist_items:
741 // create child object in lazyman fashion to avoid
742 // ownership problem with firefox
743 if( ! playlistItemsObj )
745 NPN_CreateObject(_instance, RuntimeNPClass<
746 LibvlcPlaylistItemsNPObject>::getClass());
747 OBJECT_TO_NPVARIANT(NPN_RetainObject(playlistItemsObj), result);
748 return INVOKERESULT_NO_ERROR;
754 return INVOKERESULT_GENERIC_ERROR;
757 const NPUTF8 * const LibvlcPlaylistNPObject::methodNames[] =
766 "clear", /* deprecated */
767 "removeItem", /* deprecated */
769 COUNTNAMES(LibvlcPlaylistNPObject,methodCount,methodNames);
771 enum LibvlcPlaylistNPObjectMethodIds
775 ID_playlist_playItem,
776 ID_playlist_togglepause,
781 ID_playlist_removeitem
784 RuntimeNPObject::InvokeResult
785 LibvlcPlaylistNPObject::invoke(int index, const NPVariant *args,
786 uint32_t argCount, NPVariant &result)
788 /* is plugin still running */
789 if( isPluginRunning() )
791 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
792 libvlc_exception_t ex;
793 libvlc_exception_init(&ex);
797 // XXX FIXME this needs squashing into something much smaller
798 case ID_playlist_add:
800 if( (argCount < 1) || (argCount > 3) )
801 return INVOKERESULT_NO_SUCH_METHOD;
806 if( NPVARIANT_IS_STRING(args[0]) )
808 char *s = stringValue(NPVARIANT_TO_STRING(args[0]));
811 url = p_plugin->getAbsoluteURL(s);
815 // problem with combining url, use argument
819 return INVOKERESULT_OUT_OF_MEMORY;
822 return INVOKERESULT_NO_SUCH_METHOD;
826 // grab name if available
829 if( NPVARIANT_IS_NULL(args[1]) )
833 else if( NPVARIANT_IS_STRING(args[1]) )
835 name = stringValue(NPVARIANT_TO_STRING(args[1]));
840 return INVOKERESULT_INVALID_VALUE;
845 char** ppsz_options = NULL;
847 // grab options if available
850 if( NPVARIANT_IS_NULL(args[2]) )
854 else if( NPVARIANT_IS_STRING(args[2]) )
856 parseOptions(NPVARIANT_TO_STRING(args[2]),
857 &i_options, &ppsz_options);
860 else if( NPVARIANT_IS_OBJECT(args[2]) )
862 parseOptions(NPVARIANT_TO_OBJECT(args[2]),
863 &i_options, &ppsz_options);
869 return INVOKERESULT_INVALID_VALUE;
873 int item = p_plugin->playlist_add_extended_untrusted(url, name,
874 i_options, const_cast<const char **>(ppsz_options), &ex);
877 for( int i=0; i< i_options; ++i )
879 free(ppsz_options[i]);
883 RETURN_ON_EXCEPTION(this,ex);
884 INT32_TO_NPVARIANT(item, result);
885 return INVOKERESULT_NO_ERROR;
887 case ID_playlist_play:
890 p_plugin->playlist_play(&ex);
891 RETURN_ON_EXCEPTION(this,ex);
892 VOID_TO_NPVARIANT(result);
893 return INVOKERESULT_NO_ERROR;
895 return INVOKERESULT_NO_SUCH_METHOD;
896 case ID_playlist_playItem:
897 if( (argCount == 1) && isNumberValue(args[0]) )
899 p_plugin->playlist_play_item(numberValue(args[0]),&ex);
900 RETURN_ON_EXCEPTION(this,ex);
901 VOID_TO_NPVARIANT(result);
902 return INVOKERESULT_NO_ERROR;
904 return INVOKERESULT_NO_SUCH_METHOD;
905 case ID_playlist_togglepause:
908 p_plugin->playlist_pause(&ex);
909 RETURN_ON_EXCEPTION(this,ex);
910 VOID_TO_NPVARIANT(result);
911 return INVOKERESULT_NO_ERROR;
913 return INVOKERESULT_NO_SUCH_METHOD;
914 case ID_playlist_stop:
917 p_plugin->playlist_stop(&ex);
918 RETURN_ON_EXCEPTION(this,ex);
919 VOID_TO_NPVARIANT(result);
920 return INVOKERESULT_NO_ERROR;
922 return INVOKERESULT_NO_SUCH_METHOD;
923 case ID_playlist_next:
926 p_plugin->playlist_next(&ex);
927 RETURN_ON_EXCEPTION(this,ex);
928 VOID_TO_NPVARIANT(result);
929 return INVOKERESULT_NO_ERROR;
931 return INVOKERESULT_NO_SUCH_METHOD;
932 case ID_playlist_prev:
935 p_plugin->playlist_prev(&ex);
936 RETURN_ON_EXCEPTION(this,ex);
937 VOID_TO_NPVARIANT(result);
938 return INVOKERESULT_NO_ERROR;
940 return INVOKERESULT_NO_SUCH_METHOD;
941 case ID_playlist_clear: /* deprecated */
944 p_plugin->playlist_clear(&ex);
945 RETURN_ON_EXCEPTION(this,ex);
946 VOID_TO_NPVARIANT(result);
947 return INVOKERESULT_NO_ERROR;
949 return INVOKERESULT_NO_SUCH_METHOD;
950 case ID_playlist_removeitem: /* deprecated */
951 if( (argCount == 1) && isNumberValue(args[0]) )
953 p_plugin->playlist_delete_item(numberValue(args[0]), &ex);
954 RETURN_ON_EXCEPTION(this,ex);
955 VOID_TO_NPVARIANT(result);
956 return INVOKERESULT_NO_ERROR;
958 return INVOKERESULT_NO_SUCH_METHOD;
963 return INVOKERESULT_GENERIC_ERROR;
966 // XXX FIXME The new playlist_add creates a media instance and feeds it
967 // XXX FIXME these options one at a time, so this hunk of code does lots
968 // XXX FIXME of unnecessairy work. Break out something that can do one
969 // XXX FIXME option at a time and doesn't need to realloc().
970 // XXX FIXME Same for the other version of parseOptions.
972 void LibvlcPlaylistNPObject::parseOptions(const NPString &nps,
973 int *i_options, char*** ppsz_options)
977 char *s = stringValue(nps);
982 char **options = (char **)malloc(capacity*sizeof(char *));
987 char *end = val + nps.utf8length;
990 // skip leading blanks
992 && ((*val == ' ' ) || (*val == '\t')) )
996 // skip till we get a blank character
1002 if( ('\'' == c) || ('"' == c) )
1004 // skip till end of string
1005 while( (val < end) && (*(val++) != c ) );
1011 if( nOptions == capacity )
1014 char **moreOptions = (char **)realloc(options, capacity*sizeof(char*));
1017 /* failed to allocate more memory */
1019 /* return what we got so far */
1020 *i_options = nOptions;
1021 *ppsz_options = options;
1024 options = moreOptions;
1027 options[nOptions++] = strdup(start);
1030 // must be end of string
1033 *i_options = nOptions;
1034 *ppsz_options = options;
1041 // XXX FIXME See comment at the other parseOptions variant.
1042 void LibvlcPlaylistNPObject::parseOptions(NPObject *obj, int *i_options,
1043 char*** ppsz_options)
1045 /* WARNING: Safari does not implement NPN_HasProperty/NPN_HasMethod */
1049 /* we are expecting to have a Javascript Array object */
1050 NPIdentifier propId = NPN_GetStringIdentifier("length");
1051 if( NPN_GetProperty(_instance, obj, propId, &value) )
1053 int count = numberValue(value);
1054 NPN_ReleaseVariantValue(&value);
1059 char **options = (char **)malloc(capacity*sizeof(char *));
1064 while( nOptions < count )
1066 propId = NPN_GetIntIdentifier(nOptions);
1067 if( ! NPN_GetProperty(_instance, obj, propId, &value) )
1068 /* return what we got so far */
1071 if( ! NPVARIANT_IS_STRING(value) )
1073 /* return what we got so far */
1074 NPN_ReleaseVariantValue(&value);
1078 if( nOptions == capacity )
1081 char **moreOptions = (char **)realloc(options, capacity*sizeof(char*));
1084 /* failed to allocate more memory */
1085 NPN_ReleaseVariantValue(&value);
1086 /* return what we got so far */
1087 *i_options = nOptions;
1088 *ppsz_options = options;
1091 options = moreOptions;
1094 options[nOptions++] = stringValue(value);
1096 *i_options = nOptions;
1097 *ppsz_options = options;
1104 ** implementation of libvlc subtitle object
1107 const NPUTF8 * const LibvlcSubtitleNPObject::propertyNames[] =
1113 enum LibvlcSubtitleNPObjectPropertyIds
1118 COUNTNAMES(LibvlcSubtitleNPObject,propertyCount,propertyNames);
1120 RuntimeNPObject::InvokeResult
1121 LibvlcSubtitleNPObject::getProperty(int index, NPVariant &result)
1123 /* is plugin still running */
1124 if( isPluginRunning() )
1126 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
1127 libvlc_exception_t ex;
1128 libvlc_exception_init(&ex);
1130 libvlc_media_player_t *p_md = p_plugin->getMD(&ex);
1131 RETURN_ON_EXCEPTION(this,ex);
1135 case ID_subtitle_track:
1137 /* get the current subtitle ID */
1138 int i_spu = libvlc_video_get_spu(p_md, &ex);
1139 RETURN_ON_EXCEPTION(this,ex);
1141 INT32_TO_NPVARIANT(i_spu, result);
1142 return INVOKERESULT_NO_ERROR;
1144 case ID_subtitle_count:
1146 /* get the number of subtitles available */
1147 int i_spu = libvlc_video_get_spu_count(p_md, &ex);
1148 RETURN_ON_EXCEPTION(this,ex);
1150 INT32_TO_NPVARIANT(i_spu, result);
1151 return INVOKERESULT_NO_ERROR;
1155 return INVOKERESULT_GENERIC_ERROR;
1158 RuntimeNPObject::InvokeResult
1159 LibvlcSubtitleNPObject::setProperty(int index, const NPVariant &value)
1161 /* is plugin still running */
1162 if( isPluginRunning() )
1164 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
1165 libvlc_exception_t ex;
1166 libvlc_exception_init(&ex);
1168 libvlc_media_player_t *p_md = p_plugin->getMD(&ex);
1169 RETURN_ON_EXCEPTION(this,ex);
1173 case ID_subtitle_track:
1175 if( isNumberValue(value) )
1177 /* set the new subtitle track to show */
1178 libvlc_video_set_spu(p_md, numberValue(value), &ex);
1179 RETURN_ON_EXCEPTION(this,ex);
1181 return INVOKERESULT_NO_ERROR;
1183 return INVOKERESULT_INVALID_VALUE;
1187 return INVOKERESULT_GENERIC_ERROR;
1190 const NPUTF8 * const LibvlcSubtitleNPObject::methodNames[] =
1194 COUNTNAMES(LibvlcSubtitleNPObject,methodCount,methodNames);
1196 enum LibvlcSubtitleNPObjectMethodIds
1198 ID_subtitle_description
1201 RuntimeNPObject::InvokeResult
1202 LibvlcSubtitleNPObject::invoke(int index, const NPVariant *args,
1203 uint32_t argCount, NPVariant &result)
1205 /* is plugin still running */
1206 if( isPluginRunning() )
1208 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
1209 libvlc_exception_t ex;
1210 libvlc_exception_init(&ex);
1212 libvlc_media_player_t *p_md = p_plugin->getMD(&ex);
1213 RETURN_ON_EXCEPTION(this,ex);
1217 case ID_subtitle_description:
1222 int i_spuID, i_limit, i;
1223 libvlc_track_description_t *p_spuDesc;
1225 /* get subtitles description */
1226 p_spuDesc = libvlc_video_get_spu_description(p_md, &ex);
1227 RETURN_ON_EXCEPTION(this,ex);
1229 return INVOKERESULT_GENERIC_ERROR;
1231 /* get the number of subtitle available */
1232 i_limit = libvlc_video_get_spu_count(p_md, &ex);
1233 RETURN_ON_EXCEPTION(this,ex);
1235 /* check if a number is given by the user
1236 * and get the subtitle number */
1237 if( isNumberValue(args[0]) )
1238 i_spuID = numberValue(args[0]);
1240 return INVOKERESULT_INVALID_VALUE;
1242 /* if bad number is given return invalid value */
1243 if ( ( i_spuID > ( i_limit -1 ) ) || ( i_spuID < 0 ) )
1244 return INVOKERESULT_INVALID_VALUE;
1246 /* get the good spuDesc */
1247 for( i = 0 ; i < i_spuID ; i++ )
1249 p_spuDesc = p_spuDesc->p_next;
1251 psz_name = p_spuDesc->psz_name;
1253 /* return the name of the track chosen */
1254 return invokeResultString(psz_name, result);
1256 return INVOKERESULT_NO_SUCH_METHOD;
1259 return INVOKERESULT_NO_SUCH_METHOD;
1262 return INVOKERESULT_GENERIC_ERROR;
1266 ** implementation of libvlc video object
1269 const NPUTF8 * const LibvlcVideoNPObject::propertyNames[] =
1280 enum LibvlcVideoNPObjectPropertyIds
1282 ID_video_fullscreen,
1285 ID_video_aspectratio,
1290 COUNTNAMES(LibvlcVideoNPObject,propertyCount,propertyNames);
1292 RuntimeNPObject::InvokeResult
1293 LibvlcVideoNPObject::getProperty(int index, NPVariant &result)
1295 /* is plugin still running */
1296 if( isPluginRunning() )
1298 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
1299 libvlc_exception_t ex;
1300 libvlc_exception_init(&ex);
1302 libvlc_media_player_t *p_md = p_plugin->getMD(&ex);
1303 RETURN_ON_EXCEPTION(this,ex);
1307 case ID_video_fullscreen:
1309 int val = p_plugin->get_fullscreen(&ex);
1310 RETURN_ON_EXCEPTION(this,ex);
1311 BOOLEAN_TO_NPVARIANT(val, result);
1312 return INVOKERESULT_NO_ERROR;
1314 case ID_video_height:
1316 int val = libvlc_video_get_height(p_md, &ex);
1317 RETURN_ON_EXCEPTION(this,ex);
1318 INT32_TO_NPVARIANT(val, result);
1319 return INVOKERESULT_NO_ERROR;
1321 case ID_video_width:
1323 int val = libvlc_video_get_width(p_md, &ex);
1324 RETURN_ON_EXCEPTION(this,ex);
1325 INT32_TO_NPVARIANT(val, result);
1326 return INVOKERESULT_NO_ERROR;
1328 case ID_video_aspectratio:
1330 NPUTF8 *psz_aspect = libvlc_video_get_aspect_ratio(p_md, &ex);
1331 RETURN_ON_EXCEPTION(this,ex);
1333 return INVOKERESULT_GENERIC_ERROR;
1335 STRINGZ_TO_NPVARIANT(psz_aspect, result);
1336 return INVOKERESULT_NO_ERROR;
1338 case ID_video_subtitle:
1340 int i_spu = libvlc_video_get_spu(p_md, &ex);
1341 RETURN_ON_EXCEPTION(this,ex);
1342 INT32_TO_NPVARIANT(i_spu, result);
1343 return INVOKERESULT_NO_ERROR;
1347 NPUTF8 *psz_geometry = libvlc_video_get_crop_geometry(p_md, &ex);
1348 RETURN_ON_EXCEPTION(this,ex);
1350 return INVOKERESULT_GENERIC_ERROR;
1352 STRINGZ_TO_NPVARIANT(psz_geometry, result);
1353 return INVOKERESULT_NO_ERROR;
1355 case ID_video_teletext:
1357 int i_page = libvlc_video_get_teletext(p_md, &ex);
1358 RETURN_ON_EXCEPTION(this,ex);
1359 INT32_TO_NPVARIANT(i_page, result);
1360 return INVOKERESULT_NO_ERROR;
1364 return INVOKERESULT_GENERIC_ERROR;
1367 RuntimeNPObject::InvokeResult
1368 LibvlcVideoNPObject::setProperty(int index, const NPVariant &value)
1370 /* is plugin still running */
1371 if( isPluginRunning() )
1373 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
1374 libvlc_exception_t ex;
1375 libvlc_exception_init(&ex);
1377 libvlc_media_player_t *p_md = p_plugin->getMD(&ex);
1378 RETURN_ON_EXCEPTION(this,ex);
1382 case ID_video_fullscreen:
1384 if( ! NPVARIANT_IS_BOOLEAN(value) )
1386 return INVOKERESULT_INVALID_VALUE;
1389 int val = NPVARIANT_TO_BOOLEAN(value);
1390 p_plugin->set_fullscreen(val, &ex);
1391 RETURN_ON_EXCEPTION(this,ex);
1392 return INVOKERESULT_NO_ERROR;
1394 case ID_video_aspectratio:
1396 char *psz_aspect = NULL;
1398 if( ! NPVARIANT_IS_STRING(value) )
1400 return INVOKERESULT_INVALID_VALUE;
1403 psz_aspect = stringValue(NPVARIANT_TO_STRING(value));
1406 return INVOKERESULT_GENERIC_ERROR;
1409 libvlc_video_set_aspect_ratio(p_md, psz_aspect, &ex);
1411 RETURN_ON_EXCEPTION(this,ex);
1413 return INVOKERESULT_NO_ERROR;
1415 case ID_video_subtitle:
1417 if( isNumberValue(value) )
1419 libvlc_video_set_spu(p_md, numberValue(value), &ex);
1420 RETURN_ON_EXCEPTION(this,ex);
1422 return INVOKERESULT_NO_ERROR;
1424 return INVOKERESULT_INVALID_VALUE;
1428 char *psz_geometry = NULL;
1430 if( ! NPVARIANT_IS_STRING(value) )
1432 return INVOKERESULT_INVALID_VALUE;
1435 psz_geometry = stringValue(NPVARIANT_TO_STRING(value));
1438 return INVOKERESULT_GENERIC_ERROR;
1441 libvlc_video_set_crop_geometry(p_md, psz_geometry, &ex);
1443 RETURN_ON_EXCEPTION(this,ex);
1445 return INVOKERESULT_NO_ERROR;
1447 case ID_video_teletext:
1449 if( isNumberValue(value) )
1451 libvlc_video_set_teletext(p_md, numberValue(value), &ex);
1452 RETURN_ON_EXCEPTION(this,ex);
1454 return INVOKERESULT_NO_ERROR;
1456 return INVOKERESULT_INVALID_VALUE;
1460 return INVOKERESULT_GENERIC_ERROR;
1463 const NPUTF8 * const LibvlcVideoNPObject::methodNames[] =
1468 COUNTNAMES(LibvlcVideoNPObject,methodCount,methodNames);
1470 enum LibvlcVideoNPObjectMethodIds
1472 ID_video_togglefullscreen,
1473 ID_video_toggleteletext
1476 RuntimeNPObject::InvokeResult
1477 LibvlcVideoNPObject::invoke(int index, const NPVariant *args,
1478 uint32_t argCount, NPVariant &result)
1480 /* is plugin still running */
1481 if( isPluginRunning() )
1483 VlcPlugin* p_plugin = getPrivate<VlcPlugin>();
1484 libvlc_exception_t ex;
1485 libvlc_exception_init(&ex);
1487 libvlc_media_player_t *p_md = p_plugin->getMD(&ex);
1488 RETURN_ON_EXCEPTION(this,ex);
1492 case ID_video_togglefullscreen:
1495 p_plugin->toggle_fullscreen(&ex);
1496 RETURN_ON_EXCEPTION(this,ex);
1497 VOID_TO_NPVARIANT(result);
1498 return INVOKERESULT_NO_ERROR;
1500 return INVOKERESULT_NO_SUCH_METHOD;
1501 case ID_video_toggleteletext:
1504 libvlc_toggle_teletext(p_md, &ex);
1505 RETURN_ON_EXCEPTION(this,ex);
1506 VOID_TO_NPVARIANT(result);
1507 return INVOKERESULT_NO_ERROR;
1509 return INVOKERESULT_NO_SUCH_METHOD;
1511 return INVOKERESULT_NO_SUCH_METHOD;
1514 return INVOKERESULT_GENERIC_ERROR;