当前位置: 代码迷 >> Android >> VLC Android摄制视频
  详细解决方案

VLC Android摄制视频

热度:336   发布时间:2016-05-01 11:25:32.0
VLC Android录制视频

在这篇文章《vlc android的编译及截图,录制视频等功能》里,我找到了用vlc实现Android版本截图的功能,但是录制视频的功能是不正确的。

其录制视频的方法,对vlc底层的开始录制和结束录制,都不能很好的控制。

鉴于此,我们对vlc代码进行修改,编译。

参见http://patches.videolan.org/patch/606/


diff --git a/include/vlc/libvlc_events.h b/include/vlc/libvlc_events.hindex 2cfedbf..25a16ea 100644--- a/include/vlc/libvlc_events.h+++ b/include/vlc/libvlc_events.h@@ -72,6 +72,8 @@  enum libvlc_event_e {     libvlc_MediaPlayerSnapshotTaken,     libvlc_MediaPlayerLengthChanged,     libvlc_MediaPlayerVout,+    libvlc_MediaPlayerRecordableChanged,+    libvlc_MediaPlayerRecordingFinished,      libvlc_MediaListItemAdded=0x200,     libvlc_MediaListWillAddItem,@@ -165,6 +167,14 @@  typedef struct libvlc_event_t         } media_player_pausable_changed;         struct         {+            int new_recordable;+        } media_player_recordable_changed;+        struct+        {+            char *psz_filename;+        } media_player_recording_finished;+        struct+        {             int new_count;         } media_player_vout; diff --git a/include/vlc/libvlc_media_player.h b/include/vlc/libvlc_media_player.hindex aefef02..8ddef37 100644--- a/include/vlc/libvlc_media_player.h+++ b/include/vlc/libvlc_media_player.h@@ -1628,6 +1628,121 @@  LIBVLC_API int libvlc_audio_set_delay( libvlc_media_player_t *p_mi, int64_t i_de  /** @} audio */ +/**+ * Can the media player record the current media?+ *+ * Media must be buffering or playing before it can be recorded.+ *+ * The media player event manager will emit a libvlc_MediaPlayerRecordableChanged event+ * when the recordable state changes after starting media playback. The event data will+ * describe the new recordable state, so invocation of this API method is not strictly+ * necessary to determine when recording can be started.+ *+ * A libvlc_MediaPlayerRecordableChanged event will not be emitted if the media is+ * stopped (notified by a libvlc_MediaPlayerStoppedEvent) or finishes normally (notified+ * by a libvlc_MediaPlayerFinished event).+ *+ * A calling application should therefore register an event callback for those events+ * so that it may query the new recordable state and manage recording at the appropriate+ * time.+ *+ * \param p_mi media player+ * \return true if the media player can record, false if it can not+ * \version LibVLC 2.1.0 or later+ */+LIBVLC_API bool libvlc_media_player_is_recordable( libvlc_media_player_t *p_mi );++/**+ * Is the current media being recorded?+ *+ * \param p_mi media player+ * \return true if recording, false if not+ * \version LibVLC 2.1.0 or later+ */+LIBVLC_API bool libvlc_media_player_is_recording( libvlc_media_player_t *p_mi );++/**+ * Start recording the current media.+ *+ * Media must be buffering or playing before it can be recorded. A calling application+ * can begin recording immediately on receipt of a libvlc_MediaPlayerRecordableChanged+ * event sent via the media player event manager (if recording is possible for the+ * currently playing media), and any time thereafter until the media stops.+ *+ * Media will be saved to the file path denoted by the psz_filename parameter if it is+ * supplied. Any such supplied filename should not include a file extension as the+ * correct file extension will automatically be appended when the file is created. This+ * filename may denote a full path name, but each directory in the path must already+ * exist or recording will silently fail. If the calling application chooses to specify+ * the filename then it is the responsibility of that application to take account of+ * this and itself make sure any needed directories are created.+ *+ * Alternatively, a calling application need not supply a filename and so instead let+ * vlc automatically generate a unique filename. This will cause vlc to create a new+ * file in the appropriate media directory for the user - for example "~/Videos". The+ * actual filename used will be sent in an event when the recording is complete.+ *+ * When recording has finished and the new file has been completely saved, a+ * libvlc_MediaPlayerRecordingFinished event will be sent via the media player event+ * manager. The event data will contain the filename of the newly recorded file - this+ * will either be the filename as specified by the calling application or a filename+ * generated by vlc if the application did not supply a filename. In either case, this+ * filename will include the automatically appended file extension.+ *+ * The saved media file will not be immediately available or visible until recording+ * has completely finished and the libvlc_MediaPlayerRecordingFinished event has been+ * received, or the media has stopped or finished normally.+ *+ * Recording can be stopped and started on-the-fly once the recordable state is set;+ * each time recording is stopped and restarted a new file will be created so a calling+ * application should take care to provide unique filenames, or defer to vlc to create+ * unique filenames.+ *+ * Recording will be stopped when the media stops playing, and must be explicitly+ * started again to restart recording, i.e. the recording state is not automatically+ * preserved when playing media subsequently.+ *+ * Media player functionailty such as next/previous chapter, set time or position and+ * so on are ineffective when recording is enabled. However, pausing the media is+ * possible and will pause the recording; unpausing the media will resume playback and+ * recording.+ *+ * Recording of the primary media or sub-items is possible.+ *+ * \param p_mi media player+ * \param psz_filename name of the file to save the media to, not including any file extension,+ *                     or NULL if vlc should generate the filename automatically+ * \return 0 if recording was started, -1 on error+ * \version LibVLC 2.1.0 or later+ */+LIBVLC_API int libvlc_media_player_record_start( libvlc_media_player_t *p_mi, const char *psz_filename );++/**+ * Stop recording the current media.+ *+ * This method requests that the recording stop, and will return immediately. Recording+ * will not stop immediately.+ *+ * When the recording actually stops some short time later and the new file has+ * finished being written, a libvlc_MediaPlayerRecordingFinished event will be sent via+ * the media player event manager. The newly recorded file will not be visible or+ * available until after this event has been sent.+ *+ * The event data will contain the full name of the file that was created. The filename+ * will either be that as was specified by the calling application on invoking+ * libvlc_media_player_record_start(), or the filename that vlc automatically generated+ * if the calling application did not supply its own filename. In either case the+ * filename will contain the automatically appended file extension.+ *+ * There is no need to invoke this method to stop the recording if the media is stopped+ * or finishes playing normally.+ *+ * \param p_mi media player+ * \return 0 if recording was stopped, -1 on error+ * \version LibVLC 2.1.0 or later+ */+LIBVLC_API int libvlc_media_player_record_stop( libvlc_media_player_t *p_mi );+ /** @} media_player */  # ifdef __cplusplusdiff --git a/lib/event.c b/lib/event.cindex c71a48a..7ef4abd 100644--- a/lib/event.c+++ b/lib/event.c@@ -279,6 +279,8 @@  static const event_name_t event_list[] = {     DEF(MediaPlayerSnapshotTaken)     DEF(MediaPlayerLengthChanged)     DEF(MediaPlayerVout)+    DEF(MediaPlayerRecordableChanged)+    DEF(MediaPlayerRecordingFinished)      DEF(MediaListItemAdded)     DEF(MediaListWillAddItem)diff --git a/lib/libvlc.sym b/lib/libvlc.symindex 42dad5c..3ff67ef 100644--- a/lib/libvlc.sym+++ b/lib/libvlc.sym@@ -137,6 +137,8 @@  libvlc_media_player_get_title libvlc_media_player_get_title_count libvlc_media_player_get_xwindow libvlc_media_player_has_vout+libvlc_media_player_is_recordable+libvlc_media_player_is_recording libvlc_media_player_is_seekable libvlc_media_player_is_playing libvlc_media_player_new@@ -146,6 +148,8 @@  libvlc_media_player_set_pause libvlc_media_player_pause libvlc_media_player_play libvlc_media_player_previous_chapter+libvlc_media_player_record_start+libvlc_media_player_record_stop libvlc_media_player_release libvlc_media_player_retain libvlc_media_player_set_agldiff --git a/lib/media_player.c b/lib/media_player.cindex a41b8c7..6573197 100644--- a/lib/media_player.c+++ b/lib/media_player.c@@ -64,6 +64,10 @@  input_pausable_changed( vlc_object_t * p_this, char const * psz_cmd,                         vlc_value_t oldval, vlc_value_t newval,                         void * p_userdata ); static int+input_recordable_changed( vlc_object_t *p_this, char const *psz_cmd,+                          vlc_value_t oldval, vlc_value_t newval,+                          void *p_userdata );+static int input_event_changed( vlc_object_t * p_this, char const * psz_cmd,                      vlc_value_t oldval, vlc_value_t newval,                      void * p_userdata );@@ -72,6 +76,10 @@  static int snapshot_was_taken( vlc_object_t *p_this, char const *psz_cmd,                     vlc_value_t oldval, vlc_value_t newval, void *p_data ); +static int+file_recording_finished( vlc_object_t *p_this, char const *psz_cmd,+                         vlc_value_t oldval, vlc_value_t newval, void *p_data );+ static void libvlc_media_player_destroy( libvlc_media_player_t *p_mi );  /*@@ -132,6 +140,8 @@  static void release_input_thread( libvlc_media_player_t *p_mi, bool b_input_abor                      input_seekable_changed, p_mi );     var_DelCallback( p_input_thread, "can-pause",                     input_pausable_changed, p_mi );+    var_DelCallback( p_input_thread, "can-record",+                     input_recordable_changed, p_mi );     var_DelCallback( p_input_thread, "intf-event",                      input_event_changed, p_mi ); @@ -227,6 +237,25 @@  input_pausable_changed( vlc_object_t * p_this, char const * psz_cmd, }  static int+input_recordable_changed( vlc_object_t *p_this, char const *psz_cmd,+                          vlc_value_t oldval, vlc_value_t newval,+                          void *p_userdata )+{+    VLC_UNUSED(p_this);+    VLC_UNUSED(psz_cmd);+    VLC_UNUSED(oldval);++    libvlc_media_player_t *p_mi = p_userdata;+    libvlc_event_t event;++    event.type = libvlc_MediaPlayerRecordableChanged;+    event.u.media_player_recordable_changed.new_recordable = newval.b_bool;++    libvlc_event_send( p_mi->p_event_manager, &event );+    return VLC_SUCCESS;+}++static int input_event_changed( vlc_object_t * p_this, char const * psz_cmd,                      vlc_value_t oldval, vlc_value_t newval,                      void * p_userdata )@@ -357,6 +386,23 @@  static int snapshot_was_taken(vlc_object_t *p_this, char const *psz_cmd,     return VLC_SUCCESS; } +static int file_recording_finished(vlc_object_t *p_this, char const *psz_cmd,+                                   vlc_value_t oldval, vlc_value_t newval, void *p_data )+{+    VLC_UNUSED(p_this);+    VLC_UNUSED(psz_cmd);+    VLC_UNUSED(oldval);++    libvlc_media_player_t *p_mi = p_data;+    libvlc_event_t event;++    event.type = libvlc_MediaPlayerRecordingFinished;+    event.u.media_player_recording_finished.psz_filename = newval.psz_string;++    libvlc_event_send(p_mi->p_event_manager, &event);+    return VLC_SUCCESS;+}+ static input_thread_t *find_input (vlc_object_t *obj) {     libvlc_media_player_t *mp = (libvlc_media_player_t *)obj;@@ -480,6 +526,10 @@  libvlc_media_player_new( libvlc_instance_t *instance )     var_Create (mp, "amem-set-volume", VLC_VAR_ADDRESS);     var_Create (mp, "amem-format", VLC_VAR_STRING | VLC_VAR_DOINHERIT);     var_Create (mp, "amem-rate", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT);++    var_Create (mp, "recording-finished", VLC_VAR_STRING);+    var_AddCallback (mp, "recording-finished", file_recording_finished, mp);+     var_Create (mp, "amem-channels", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT);      mp->p_md = NULL;@@ -515,6 +565,9 @@  libvlc_media_player_new( libvlc_instance_t *instance )     register_event(mp, TitleChanged);     register_event(mp, PausableChanged); +    register_event(mp, RecordableChanged);+    register_event(mp, RecordingFinished);+     register_event(mp, Vout);      /* Snapshot initialization */@@ -566,6 +619,8 @@  static void libvlc_media_player_destroy( libvlc_media_player_t *p_mi )     var_DelCallback( p_mi->p_libvlc,                      "snapshot-file", snapshot_was_taken, p_mi ); +    var_DelCallback( p_mi, "recording-finished", file_recording_finished, p_mi );+     /* No need for lock_input() because no other threads knows us anymore */     if( p_mi->input.p_thread )         release_input_thread(p_mi, true);@@ -732,12 +787,14 @@  int libvlc_media_player_play( libvlc_media_player_t *p_mi )      var_AddCallback( p_input_thread, "can-seek", input_seekable_changed, p_mi );     var_AddCallback( p_input_thread, "can-pause", input_pausable_changed, p_mi );+    var_AddCallback( p_input_thread, "can-record", input_recordable_changed, p_mi );     var_AddCallback( p_input_thread, "intf-event", input_event_changed, p_mi );      if( input_Start( p_input_thread ) )     {         unlock_input(p_mi);         var_DelCallback( p_input_thread, "intf-event", input_event_changed, p_mi );+        var_DelCallback( p_input_thread, "can-record", input_recordable_changed, p_mi );         var_DelCallback( p_input_thread, "can-pause", input_pausable_changed, p_mi );         var_DelCallback( p_input_thread, "can-seek", input_seekable_changed, p_mi );         vlc_object_release( p_input_thread );@@ -1409,3 +1466,62 @@  void libvlc_media_player_next_frame( libvlc_media_player_t *p_mi )         vlc_object_release( p_input_thread );     } }++bool libvlc_media_player_is_recordable( libvlc_media_player_t *p_mi )+{+    input_thread_t *p_input_thread;+    bool b_can_record;++    p_input_thread = libvlc_get_input_thread( p_mi );+    if( !p_input_thread )+        return false;++    b_can_record = var_GetBool( p_input_thread, "can-record" );++    vlc_object_release( p_input_thread );+    return b_can_record;+}++bool libvlc_media_player_is_recording( libvlc_media_player_t *p_mi )+{+    input_thread_t *p_input_thread;+    bool b_record;++    p_input_thread = libvlc_get_input_thread( p_mi );+    if( !p_input_thread )+        return false;++    b_record = var_GetBool( p_input_thread, "record" );++    vlc_object_release( p_input_thread );+    return b_record;+}++int libvlc_media_player_record_start( libvlc_media_player_t *p_mi, const char* psz_filename )+{+    input_thread_t *p_input_thread;++    p_input_thread = libvlc_get_input_thread( p_mi );+    if( !p_input_thread )+        return -1;++    var_SetString( p_input_thread, "input-record-path", psz_filename );+    var_SetBool( p_input_thread, "record", true );++    vlc_object_release( p_input_thread );+    return 0;+}++int libvlc_media_player_record_stop( libvlc_media_player_t *p_mi )+{+    input_thread_t *p_input_thread;++    p_input_thread = libvlc_get_input_thread( p_mi );+    if( !p_input_thread )+        return -1;++    var_SetBool( p_input_thread, "record", false );++    vlc_object_release( p_input_thread );+    return 0;+}diff --git a/modules/stream_out/record.c b/modules/stream_out/record.cindex de6d32e..40ddfea 100644--- a/modules/stream_out/record.c+++ b/modules/stream_out/record.c@@ -110,6 +110,8 @@  struct sout_stream_sys_t     int              i_id;     sout_stream_id_t **id;     mtime_t     i_dts_start;++    char *psz_record_file; };  static void OutputStart( sout_stream_t *p_stream );@@ -158,6 +160,8 @@  static int Open( vlc_object_t *p_this )     p_sys->i_dts_start = 0;     TAB_INIT( p_sys->i_id, p_sys->id ); +    p_sys->psz_record_file = NULL;+     return VLC_SUCCESS; } @@ -172,6 +176,19 @@  static void Close( vlc_object_t * p_this )     if( p_sys->p_out )         sout_StreamChainDelete( p_sys->p_out, p_sys->p_out ); +    if( p_sys->psz_record_file ) {+        for( vlc_object_t *p_mp = p_stream->p_parent; p_mp; p_mp = p_mp->p_parent )+        {+            if( var_Type( p_mp, "recording-finished" ) )+            {+                var_SetString( p_mp, "recording-finished", p_sys->psz_record_file );+                break;+            }+        }++        free( p_sys->psz_record_file );+    }+     TAB_CLEAN( p_sys->i_id, p_sys->id );     free( p_sys->psz_prefix );     free( p_sys );@@ -352,7 +369,10 @@  static int OutputNew( sout_stream_t *p_stream,     }      if( psz_file && psz_extension )+    {+        p_sys->psz_record_file = strdup( psz_file );         var_SetString( p_stream->p_libvlc, "record-file", psz_file );+    }      free( psz_file );     free( psz_output );diff --git a/src/input/var.c b/src/input/var.cindex 9613fe2..04f33b9 100644--- a/src/input/var.c+++ b/src/input/var.c@@ -210,6 +210,9 @@  void input_ControlVarInit ( input_thread_t *p_input )     text.psz_string = _("Subtitles Track");     var_Change( p_input, "spu-es", VLC_VAR_SETTEXT, &text, NULL ); +    /* ES Out */+    var_Create( p_input, "input-record-path", VLC_VAR_STRING | VLC_VAR_DOINHERIT );+     /* Special read only objects variables for intf */     var_Create( p_input, "bookmarks", VLC_VAR_STRING | VLC_VAR_DOINHERIT );

再此基础上,添加jni的接口,编译,然后给java应用层调用,就可以实现Android版的vlc录制视频了。

  相关解决方案