分段播放的flash播放器

效果:html

視頻分段好處顯而易見,就是節省流量,由於看視頻不少時候都不會看完,還有不少時候是跳着看的。還有的時候也許用戶暫停視頻出去買東西了。。。nginx

本文不討論flash rtmp直播流,例子用的是普通的http流,視頻7分鐘一段,播放至當前視頻的90%時開始加載下一段。git

原理很簡單,就是僞視頻流和對播放時間的判斷,還有一些小的細節。github

關於視頻僞流技術(pseudo streaming)能夠參考flowplayer的這篇介紹http://flash.flowplayer.org/plugins/streaming/pseudostreaming.html,簡單說就是這個東西能夠讓用戶不用等視頻緩衝到後面就能夠點擊後面沒緩衝的位置播放,這點很實用。apache

1.安裝服務器視頻模塊,讓服務器支持客戶端發出的對視頻相應幀的請求。個人是apache httpd服務器

LoadModule flvx_module modules/mod_flvx.so
LoadModule h264_streaming_module modules/mod_h264_streaming.soide

nginx也是在編譯的時候添加nginx對這兩個視頻格式模塊的支持參數。post

2.這樣客戶端就是能夠用有start參數的連接讓服務器傳輸從相應幀開始的視頻。ui

對於mp4,解析出this

time 0
offset 217027
time 2.333
offset 361576
time 11.733
offset 1010319

。。。。

請求格式start= _frameInfo[frame].time

flv,f4v因爲格式內沒有對相應幀的索引,因此必須用flvtool或yamdi添加,添加後解析出

filepositions  5388,36263,65370,93925,。。。。。。

times  0.065,0.545,1.025,1.505,1.985,。。。。。。。。

請求格式start= _frameInfo.filepositions[frame])

as

main.as

  1 package 
  2 {
  3     import flash.display.*;
  4     import ui.*;
  5     import flash.events.*;
  6     import core.*;
  7     import flash.geom.*;
  8 
  9     public class Youtube_player extends MovieClip
 10     {
 11         private var W:int = 360;
 12         private var H:int = 266;
 13         private var fullscreen:Boolean = true;
 14         private var video:String;
 15         private var thumbnail:String;
 16         private var playing:Boolean = false;
 17         private var load_relate:Boolean = false;
 18         private var autoplay:Boolean = true;
 19         private var cur_vl:VideoLoader;//當前videoloader
 20         private var loader_arr:Array;
 21         private var loading:Boolean = false;//是否在加載緩衝
 22         private var fs:Boolean = false;
 23         private var _nav_bar;
 24         private var videoHeight:int;
 25         private var videoWidth:int;
 26         private var video_index:int = 0;
 27         private var duration:int = 1232;//總時長
 28         private var part_duration:int = 420;//每段長度
 29         private var cur_total_time:int = 0;
 30         private var v:VideoLoader;//視頻操做的封裝類
 31         private var cur_i:int;//當前視頻index
 32         private var isPlay:Boolean;
 33 
 34         var arr=['http://localhost/twitter/videos/1.mp4?'+new Date().getTime(),'http://localhost/twitter/videos/3.mp4?'+new Date().getTime(),'http://localhost/twitter/videos/4.mp4?'+new Date().getTime()];
 35         //var arr=['http://localhost/twitter/videos/2.flv?'+new Date().getTime(),'http://localhost/twitter/videos/3.flv?'+new Date().getTime(),'http://localhost/twitter/videos/4.flv?'+new Date().getTime()];
 36 
 37         public function Youtube_player()
 38         {
 39             _load_config(this.loaderInfo);
 40             _init();
 41             _init_ui(W,H);
 42             _init_events();
 43         }
 44         function resizeEvent(e:Event):void
 45         {
 46             _init_ui(stage.stageWidth, stage.stageHeight);
 47             size_screen(W,H);//重繪ui
 48         }
 49         private function add_video_listener(i:int)
 50         {
 51             var vl:VideoLoader = loader_arr[i] as VideoLoader;
 52             var vl1:VideoLoader = null;
 53             if (i<arr.length-1)
 54             {
 55                 vl1 = loader_arr[i + 1] as VideoLoader;
 56             }
 57             vl.volume(0);
 58             if (! vl.hasEventListener("playProgress"))
 59             {
 60                 vl.addEventListener("playProgress",function(){
 61                     if(vl.status=="NetStream.Seek.Notify")
 62                     buffering.visible = true;
 63                     else
 64                     buffering.visible = false;
 65                     var vl1_buffer_progress=0;
 66                     var a=vl.isFlv?cur_total_time+vl.videoTime:cur_total_time+vl.videoTime+vl.start;
 67                     if(vl.videoTime>1){
 68                         nav.progress_line.x=a/duration*nav.bar.width;
 69                         nav.notify.text=formatTime(a)+" / "+formatTime(duration);
 70                     }
 71                     if(vl1!=null&&!isNaN(vl1.bufferProgress)&&vl1.bufferProgress!=0){
 72                         vl1_buffer_progress=vl1.bufferProgress*vl1.totalTime;
 73                     }
 74                     var vl_buffer_progress=0;
 75                     if(!isNaN(vl.bufferProgress)){
 76                         vl_buffer_progress=vl.isFlv?vl.bufferProgress*(vl.totalTime-vl.start):vl.bufferProgress*vl.totalTime;
 77                     }
 78                     if(!isNaN(vl_buffer_progress)&&!isNaN(vl1_buffer_progress)){
 79                         if(i==arr.length-1||vl_buffer_progress!=0&&vl.isPlay){
 80                             nav.progressBar.width=(vl.start+vl1_buffer_progress+cur_total_time+vl_buffer_progress)/duration*nav.bar.width;
 81                         }
 82                     }
 83                     var prop=vl.isMp4?(loader_arr[cur_i].start+loader_arr[cur_i].videoTime)/(loader_arr[cur_i].totalTime+loader_arr[cur_i].start):vl.playProgress;
 84                     if(!loading&&prop>0.9&&prop<1){//播放到90%加載下一段
 85                     if(vl1!=null){
 86                         if(vl.isMp4){
 87                             vl1.metaData_loaded=false;
 88                             vl1.start=0;
 89                         }
 90                         vl1.load();
 91                         screen.setChildIndex(vl1.content,0);
 92                         vl1.pauseVideo();
 93                         loading=true;
 94                     }
 95                 }
 96                 });
 97             }
 98             if (! vl.hasEventListener("videoComplete"))
 99             {
100                 vl.addEventListener("videoComplete",function(){
101                     if(vl1!=null){
102                         screen.setChildIndex(vl1.content,screen.numChildren -1);
103                         if(!vl1.isPlay)
104                         vl1.pauseVideo();
105                         cur_i=i+1;
106                         cur_total_time=vl.isFlv?cur_total_time+vl.totalTime:cur_total_time+vl.totalTime+vl.start;
107                         loading=false;
108                     }
109                     i++;
110                     if(i<arr.length)
111                     add_video_listener(i);
112                     else{
113                         nav.progressBar.visible=false;
114                         nav.playingBar.visible=false;
115                         new Recommend(stage,W,H);
116                     }
117                     });
118                }
119             }
120             private function _init()
121             {
122             v = new VideoLoader(arr[0],{name:"myVideo0",container:screen,width:360,height:240});
123             for (var i=1,len=arr.length; i<len; i++)
124             {
125                 new VideoLoader(arr[i],{name:"myVideo" + i,container:screen,width:360,height:240});
126             }
127             loader_arr = VideoLoader.getLoader();
128             v.load();
129             cur_i = 0;
130             add_video_listener(0);//添加對每段視頻的監聽
131             v.addEventListener("videoStart",function(){
132                 size_screen(W,H);
133                 });
134             _nav_bar = new Nav_bar(nav,top_nav,overlay);
135             new Screen_size(stage,screen,top_nav.screen_size.size1,top_nav.screen_size.size2,top_nav.screen_size.size3);
136         }
137         private function progressBarEvent(e:MouseEvent):void//點擊進度條
138         {
139             var point:Number = stage.mouseX;
140             var seekpoint:int = (point / nav.bar.width) * duration;
141             var index=Math.floor(seekpoint/part_duration);
142             var load_video:VideoLoader = loader_arr[index] as VideoLoader;
143             var load_video_buffer = loader_arr[index].bufferProgress * loader_arr[index].totalTime;
144             var seek_time = load_video.isFlv ? seekpoint % part_duration:seekpoint % part_duration - load_video.start;
145             if (isNaN(load_video_buffer)||(seekpoint%part_duration<load_video.start||seekpoint%part_duration>load_video.start+load_video_buffer))
146             {
147                 if (cur_i!=index)
148                 {
149                     add_video_listener(index);
150                     load_video.load(seekpoint % part_duration);
151                     cur_total_time = cur_total_time + part_duration * (index - cur_i);
152                     if (cur_i<loader_arr.length-1)
153                     {
154                         if (Math.abs(cur_i-index)>1)
155                         {
156                             loader_arr[cur_i + 1].netStream.close();
157                         }
158                     }
159                     loader_arr[cur_i].netStream.close();
160                     screen.setChildIndex(loader_arr[index].content,screen.numChildren -1);
161                 }
162                 else
163                 {
164                     load_video.load(seekpoint % part_duration);
165                     screen.setChildIndex(load_video.content,screen.numChildren -1);
166                 }
167             }
168             else
169             {
170                 if (cur_i==index)
171                 {
172                     load_video.seekVideo(seek_time);
173                 }
174                 else
175                 {
176                     add_video_listener(index);
177                     loader_arr[index].pauseVideo();
178                     loader_arr[index].seekVideo(seek_time);
179                     loader_arr[cur_i].netStream.close();
180                     cur_total_time = cur_total_time + part_duration * (index - cur_i);
181                     screen.setChildIndex(loader_arr[index].content,screen.numChildren -1);
182                     loading = false;
183                 }
184             }
185             cur_i = index;
186         }
187         private function _init_ui(W,H)
188         {
189             stage.scaleMode = StageScaleMode.NO_SCALE;
190             stage.align = StageAlign.TOP_LEFT;
191             new Button(nav.playButton);
192             new Button(nav.pauseButton);
193             new Button(nav.fullscreen);
194             new Button(nav.volumn_btn);
195             new Button(nav.mute);
196             this.W = W;
197             this.H = H;
198             background.x = screen.x = overlay.x = 0;
199             background.y = screen.y = overlay.y = 0;
200             background.width = overlay.width = W;
201             background.height = overlay.height = H;
202             overlay.alpha = 0;
203             buffering.x = (W - buffering.width) * .5;
204             buffering.y = (H - buffering.height) * .5;
205             nav.y = H - 26;
206             top_nav.width = nav.bar.width = W;
207             top_nav.time.x = W - 70;
208             nav.pauseButton.y = nav.playButton.y;
209             nav.progressBar.x = nav.playingBar.x = 0;
210             nav.progressBar.y = nav.playingBar.y = nav.bar.y - 10;
211             nav.seeker_time.visible = false;
212             nav.seeker_time.y = nav.progressBar.y - 20;
213             nav.seeker_time.x = 0;
214             if (fullscreen)
215             {
216                 nav.fullscreen.x = nav.bar.width - nav.fullscreen.width - 20;
217             }
218             else
219             {
220                 nav.fullscreen.visible = false;
221             }
222             nav.playingBar.width = nav.bar.width;
223             nav.notify.x = nav.volumeBar.x + nav.volumeBar.width + 10;
224             nav.progressBar.width = W;
225             if (autoplay)
226             {
227                 nav.playButton.visible = false;
228             }
229             else
230             {
231                 nav.pauseButton.visible = false;
232             }
233             nav.volumn_btn.x = nav.volumeBar.x + 36;
234             nav.y = overlay.height - 26;
235             nav.progress_line.x = top_nav.y = -30;
236             nav.seeker_time.y = nav.progressBar.y - 20;
237             nav.seeker_time.x = 0;
238             nav.hover_line.y = nav.progress_line.y = nav.progressBar.y;
239             nav.setChildIndex(nav.progress_line,nav.numChildren-1);
240             nav.setChildIndex(nav.hover_line,nav.numChildren-1);
241             nav.hover_line.visible = false;
242             nav.hover_line.buttonMode = true;
243             nav.progress_line.buttonMode = true;
244             nav.progressBar.buttonMode = true;
245             nav.playingBar.buttonMode = true;
246         }
247         private function _init_events()
248         {
249             stage.addEventListener(Event.RESIZE, resizeEvent);
250             stage.addEventListener(Event.FULLSCREEN, resizeEvent);
251             stage.addEventListener(MouseEvent.MOUSE_MOVE,_nav_bar.nav_show);
252             nav.fullscreen.addEventListener(MouseEvent.CLICK, fullscreenEvent);
253             nav.progressBar.addEventListener(MouseEvent.MOUSE_OVER, seeker_time_show);
254             nav.playingBar.addEventListener(MouseEvent.MOUSE_OVER, seeker_time_show);
255             nav.progressBar.addEventListener(MouseEvent.MOUSE_MOVE, seeker_time_show);
256             nav.playingBar.addEventListener(MouseEvent.MOUSE_MOVE, seeker_time_show);
257             nav.progressBar.addEventListener(MouseEvent.MOUSE_OUT, seeker_time_hide);
258             nav.playingBar.addEventListener(MouseEvent.MOUSE_OUT, seeker_time_hide);
259             nav.playingBar.addEventListener(MouseEvent.CLICK, progressBarEvent);
260             nav.progressBar.addEventListener(MouseEvent.CLICK, progressBarEvent);
261             nav.playButton.addEventListener(MouseEvent.CLICK, playOrpauseEvent);
262             nav.pauseButton.addEventListener(MouseEvent.CLICK, playOrpauseEvent);
263             nav.volumeBar.addEventListener(MouseEvent.CLICK, volume_bar_click);
264             nav.mute.addEventListener(MouseEvent.CLICK, volumeEvent);
265             nav.volumn_btn.addEventListener(MouseEvent.MOUSE_DOWN, volumn_btn_mouse_down);
266             stage.addEventListener(MouseEvent.MOUSE_UP, volumn_btn_mouse_up);
267             top_nav.screen_size.size3.addEventListener(MouseEvent.CLICK,size3_handler);
268         }
269         private function size3_handler(e:MouseEvent):void
270         {
271             var proportion:Number = W / H;
272             var videoproportion:Number = loader_arr[cur_i].content.width / loader_arr[cur_i].content.height;
273             if (videoproportion >= proportion)
274             {
275                 screen.width = W;
276                 screen.height = W / videoproportion;
277             }
278             else
279             {
280                 screen.width = H * videoproportion;
281                 screen.height = H;
282             }
283             screen.x = (stage.stageWidth -screen.width) *0.5;
284             screen.y = (stage.stageHeight - screen.height) *0.5;
285         }
286         private function setVolume(newVolume:Number):void
287         {
288             loader_arr[cur_i].volume(newVolume);
289             nav.mute.gotoAndStop((newVolume > 0) ? 1 : 2);
290         }
291         private function volumeEvent(event:MouseEvent):void
292         {
293             setVolume(loader_arr[cur_i].mute());
294         }
295         private function volume_bar_click(event:MouseEvent):void
296         {
297             var volume = event.localX / 140;
298             nav.mute.gotoAndStop((volume > 0) ? 1 : 2);
299             loader_arr[cur_i].volume(new Number((volume.toFixed(2))));
300             nav.volumn_btn.x = nav.volumeBar.x + event.localX;
301             event.stopPropagation();
302         }
303         private function volumn_btn_mouse_down(e:Event)
304         {
305             var rect:Rectangle = new Rectangle(nav.volumeBar.x,nav.volumn_btn.y);
306             rect.width = nav.volumeBar.width - nav.volumn_btn.width;
307             rect.height = 0;
308             e.target.startDrag(false,rect);
309             nav.volumn_btn.addEventListener(MouseEvent.MOUSE_MOVE, volumn_btn_mouse_move);
310             e.stopPropagation();
311         }
312         private function volumn_btn_mouse_move(event:MouseEvent):void
313         {
314             var volume=(nav.volumn_btn.x-nav.volumeBar.x)/(140-nav.volumn_btn.width);
315             nav.mute.gotoAndStop((volume > 0) ? 1 : 2);
316             loader_arr[cur_i].volume(new Number((volume.toFixed(2))));
317             event.stopPropagation();
318         }
319         private function volumn_btn_mouse_up(event:MouseEvent):void
320         {
321             nav.volumn_btn.stopDrag();
322             nav.volumn_btn.removeEventListener(MouseEvent.MOUSE_MOVE, volumn_btn_mouse_move);
323             event.stopPropagation();
324         }
325         private function playOrpauseEvent(event:MouseEvent):void
326         {
327             isPlay = loader_arr[cur_i].pauseVideo();
328             nav.playButton.visible = isPlay;
329             nav.pauseButton.visible = ! isPlay;
330         }
331         private function _load_config(li:LoaderInfo)
332         {
333             video = this.loaderInfo.parameters.video;
334             thumbnail = this.loaderInfo.parameters.thumbnail;
335             autoplay = this.loaderInfo.parameters.autoplay == 1;
336             load_relate = this.loaderInfo.parameters.load_relate;
337             W=this.loaderInfo.parameters.width;
338             H=this.loaderInfo.parameters.height;
339         }
340         private function size_screen(W:int,H:int):void
341         {
342             top_nav.time.x = W - 70;
343             top_nav.screen_size.size2.gotoAndStop(1);
344             top_nav.screen_size.size3.gotoAndStop(1);
345             top_nav.screen_size.size1.gotoAndStop(1);
346             top_nav.screen_size.size2.graphics.clear();
347             top_nav.screen_size.size3.graphics.clear();
348             top_nav.screen_size.size1.graphics.clear();
349             if (fs)
350             {
351                 top_nav.visible = true;
352                 var proportion:Number = W / H;
353                 var videoproportion:Number = loader_arr[cur_i].content.width / loader_arr[cur_i].content.height;
354                 if (videoproportion >= proportion)
355                 {//<= (H / W)
356                     screen.width = W;
357                     screen.height = W / videoproportion;
358                 }
359                 else
360                 {
361                     screen.width = H * videoproportion;
362                     screen.height = H;
363                 }
364                 top_nav.screen_size.size3.gotoAndStop(2);
365             }
366             else
367             {
368                 top_nav.visible = false;
369                 screen.width = loader_arr[cur_i].content.width;
370                 screen.height = loader_arr[cur_i].content.height;
371             }
372             screen.x = (stage.stageWidth -screen.width) *0.5;
373             screen.y = (stage.stageHeight - screen.height) *0.5;
374         }
375         public function fullscreenEvent(e:Event):void
376         {
377             if (stage.displayState == StageDisplayState.FULL_SCREEN)
378             {
379                 fs = false;
380                 stage.displayState = StageDisplayState.NORMAL;
381             }
382             else
383             {
384                 fs = true;
385                 stage.displayState = StageDisplayState.FULL_SCREEN;
386             }
387         }
388         private function seeker_time_show(e:MouseEvent):void
389         {
390             var target_x = stage.mouseX;
391             nav.hover_line.x = target_x;
392             if (15<target_x&&target_x<W-15)
393             {
394                 nav.hover_line.visible = true;
395                 nav.seeker_time.visible = true;
396                 nav.seeker_time.x = target_x;
397                 nav.seeker_time.mouseon_time.text = formatTime(stage.mouseX / W * duration);
398             }
399         }
400         private function seeker_time_hide(e:MouseEvent):void
401         {
402             nav.hover_line.visible = false;
403             nav.seeker_time.visible = false;
404         }
405         private function formatTime(time:Number):String
406         {
407             if (time > 0)
408             {
409                 var integer:String = String((time / 60) >> 0);
410                 var decimal:String = String((time % 60) >> 0);
411                 return ((integer.length < 2) ? "0" + integer : integer) + ":" + ((decimal.length < 2) ? "0" + decimal : decimal);
412             }
413             else
414             {
415                 return String("00:00");
416             }
417         }
418     }
419 }

VideoLoader.as

  1 package core
  2 {
  3     import flash.display.*;
  4     import flash.net.*;
  5     import flash.media.*;
  6     import flash.events.*;
  7     import flash.utils.*;
  8 
  9     public class VideoLoader extends Shape
 10     {
 11         private var nc:NetConnection;
 12         private var _ns:NetStream;
 13         private var st:SoundTransform;
 14         private var _url:String;
 15         private var vars:Object;
 16         private var v:Video;
 17         private var _status:String;
 18         private var togglepause:Boolean = false;
 19         private var _duration:int;
 20         private static var _loader = [];
 21         private var _frameInfo:Object;
 22         private var _start:Number = 0;
 23         private var _metaData_loaded:Boolean = false;
 24         private var _volcache:Number = 0;
 25         private var _isMp4:Boolean = false;
 26         private var _isFlv:Boolean = false;
 27 
 28         public function VideoLoader(url:String,vars:Object=null)
 29         {
 30             nc = new NetConnection  ;
 31             nc.connect(null);
 32             _ns = new NetStream(nc);
 33             _ns.addEventListener(NetStatusEvent.NET_STATUS,nsEvent);
 34             _ns.client = {onMetaData:this._metaDataHandler};
 35             _ns.bufferTime = 1;
 36             st = new SoundTransform  ;
 37             v = new Video  ;
 38             v.attachNetStream(_ns);
 39             getVideoType(url);
 40             this._url = url;
 41             this.vars = vars;
 42             _loader.push(this);
 43             v.smoothing = true;
 44             vars.container.addChildAt(v,0);
 45         }
 46         private function nsEvent(e:NetStatusEvent):void
 47         {
 48             if ((_status != e.info.code))
 49             {
 50                 switch (e.info.code)
 51                 {
 52                     case "NetConnection.Connect.Success" :
 53                         break;
 54                     case "NetStream.Play.Stop" :
 55                         stopVideo();
 56                         break;
 57                     case "NetStream.Play.Start" :
 58                         break;
 59                 }
 60                 _status = e.info.code;
 61                 _render();
 62             }
 63         }
 64         public function mute():Number
 65         {
 66             if (_volcache)
 67             {
 68                 st.volume = _volcache;
 69                 _ns.soundTransform = st;
 70                 _volcache = 0;
 71             }
 72             else
 73             {
 74                 _volcache = st.volume;
 75                 st.volume = 0;
 76                 _ns.soundTransform = st;
 77             }
 78             return st.volume;
 79         }
 80         public function load(start:Number=0):void
 81         {
 82             //trace(_metaData_loaded);
 83             _start = start;
 84             if (nc.connected)
 85             {
 86                 if (_metaData_loaded)
 87                 {
 88                     var frame:Number = _isFlv ? get_nearest_keyframe(_start,_frameInfo.times):get_nearest_seekpoint(_start,_frameInfo);
 89                     if (_isFlv)
 90                     {
 91                         _ns.play(((_url + "&start=") + _frameInfo.filepositions[frame]));
 92                     }
 93                     else
 94                     {
 95                         _ns.play(((_url + "&start=") + _frameInfo[frame].time));
 96                     }
 97                 }
 98                 else
 99                 {
100                     _ns.play(_url);
101                 }
102             }
103             else
104             {
105                 _ns.play(_url);
106             }
107         }
108         private function _metaDataHandler(i:Object):void
109         {
110 //            for (var j=0,len=i['seekpoints'].length; j<len; j++)
111 //            {
112 //            var h = i['seekpoints'][j];
113 //            for (var k in h)
114 //            {
115 //            trace(k+"  "+h[k]);
116 //            }
117 //            }
118 //            for (var j in i['keyframes'])
119 //            {
120 //                trace(((j + "  ") + i['keyframes'][j]));
121 //            }
122             _duration = i.duration;
123             _frameInfo = _isFlv ? i['keyframes']:i['seekpoints'];
124             if (typeof vars.width == 'undefined')
125             {
126                 v.width = i.width;
127             }
128             if (typeof vars.height == 'undefined')
129             {
130                 v.height = i.height;
131             }
132             if ((_start != 0&&!_metaData_loaded))
133             {
134                 _ns.close();
135                 var frame:Number = _isFlv ? get_nearest_keyframe(_start,_frameInfo.times):get_nearest_seekpoint(_start,_frameInfo);
136                 if (_isFlv)
137                 {
138                     _ns.play(((_url + "&start=") + _frameInfo.filepositions[frame]));
139                 }
140                 else
141                 {
142                     _ns.play(((_url + "&start=") + _frameInfo[frame].time));
143                 }
144             }
145             _metaData_loaded = true;
146         }
147         private function get_nearest_seekpoint(second:Number,seekpoints)
148         {
149             var index1 = seekpoints.length - 1;
150             var index2 = seekpoints.length - 1;
151             for (var i = 0; i != seekpoints.length; i++)
152             {
153                 if (seekpoints[i]["time"] < second)
154                 {
155                     index1 = i;
156                 }
157                 else
158                 {
159                     index2 = i;
160                     break;
161                 }
162             }
163             if (((second - seekpoints[index1]["time"]) < seekpoints[index2]["time"] - second))
164             {
165                 return index1;
166             }
167             else
168             {
169                 return index2;
170             }
171         }
172         private function get_nearest_keyframe(second:Number,keytimes)
173         {
174             var index1 = 0;
175             var index2 = 0;
176             for (var i = 0; i != keytimes.length; i++)
177             {
178                 if (keytimes[i] < second)
179                 {
180                     index1 = i;
181                 }
182                 else
183                 {
184                     index2 = i;
185                     break;
186                 }
187             }
188             if (((second - keytimes[index1]) < keytimes[index2] - second))
189             {
190                 return index1;
191             }
192             else
193             {
194                 return index2;
195             }
196         }
197         private function getVideoType(url:String):void
198         {
199             var index = url.lastIndexOf(".flv");
200             if ((index != -1))
201             {
202                 _isFlv = true;
203             }
204             else
205             {
206                 _isFlv = false;
207             }
208             _isMp4 = ! _isFlv;
209         }
210         public function get isFlv():Boolean
211         {
212             return _isFlv;
213         }
214         public function get isMp4():Boolean
215         {
216             return _isMp4;
217         }
218         public function pauseVideo():Boolean
219         {
220             if (togglepause)
221             {
222                 togglepause = false;
223                 _ns.resume();
224             }
225             else
226             {
227                 togglepause = true;
228                 _ns.pause();
229             }
230             return togglepause;
231         }
232         public function volume(vol:Number)
233         {
234             st.volume = vol;
235             _ns.soundTransform = st;
236         }
237         public function stopVideo():void
238         {
239             this.dispatchEvent(new Event("videoComplete"));
240             _ns.close();
241             _metaData_loaded = false;
242         }
243         private function _render():void
244         {
245             var timer:Timer = new Timer(100);
246             var _this = this;
247             this.dispatchEvent(new Event("videoStart"));
248             var timerEvent = function(e:TimerEvent):void
249             {
250             _this.dispatchEvent(new Event("playProgress"));
251             };
252             timer.addEventListener(TimerEvent.TIMER,timerEvent);
253             timer.start();
254         }
255         public function seekVideo(point:Number):void
256         {
257             _ns.seek(point);
258         }
259         public function get videoPaused():Boolean
260         {
261             return togglepause;
262         }
263         public function get content():Video
264         {
265             return v;
266         }
267         public function get playProgress():Number
268         {
269             return _ns.time / _duration;
270         }
271         public function get bufferProgress():Number
272         {
273             return _ns ? _ns.bytesLoaded / _ns.bytesTotal:1;
274         }
275         public function get videoTime():Number
276         {
277             return _ns ? _ns.time:0;
278         }
279         public function get totalTime():Number
280         {
281             return _duration;
282         }
283         public static function getLoader():Array
284         {
285             return _loader;
286         }
287         public function get netStream():NetStream
288         {
289             return _ns;
290         }
291         public function set url(url:String):void
292         {
293             getVideoType(url);
294             _url = url;
295         }
296         public function get isPlay():Boolean
297         {
298             return !togglepause;
299         }
300         public function set metaData_loaded(metaData_loaded:Boolean):void
301         {
302             this._metaData_loaded=metaData_loaded;
303         }
304         public function get status():String
305         {
306             return _status;
307         }
308         public function get frameInfo():Object
309         {
310             return _frameInfo;
311         }
312         public function get start():Number
313         {
314             return _start;
315         }
316         public function set start(start:Number):void
317         {
318             this._start = start;
319         }
320 
321     }
322 }

由於嫌棄greensocks的loadermax裏面的videoloader太大了,本身就重寫了個。

題外話

事實上,若是開啓了優酷客戶端的加速器去網頁上看優酷,能夠看到優酷也是分了段的,也會發出1.flv?start=100那種請求。而不加速器就會是下圖的種子請求

連接相似於:

http://27.221.48.210/youku/6571EC9C9743582E25E8A83AE9/030002070250ECC0BDF242023AEBCA6C8500D8-AB84-D3E6-1355-AD3E89D20E76.flv?nk=410723804503_23592369908&ns=11949300_2673200&special=true

優酷視頻地址分析參見http://www.cnblogs.com/keygle/p/3829653.html,, http://www.cnblogs.com/zhaojunjie/p/4009192.html

能夠看出優酷是用了蘋果的m3u8.m3u8將那些5-10秒的分段小視頻經過索引組織起來,並且能夠調整適應碼率。

下面問題來了,flash怎麼解析m3u8..參見http://player.sewise.com/a/yingyongshili/20140425/7.html.

這個player實際是用了osmf和HLSProvider的m3u8解析,調用起來實際上也很簡單.這樣就大大的增長了m3u8的適用範圍。缺點就是至少增長了150k的體積。

本屌的另外一篇文章 小巧的http live streaming m3u8播放器

轉載請註明 TheViper   http://www.cnblogs.com/TheViper

相關文章
相關標籤/搜索