我正在尝试构建一个能够无缝串联播放视频URL列表的视频序列器。视频之间不能有差距;播放需要尽可能接近帧精确度。
尝试#1 我使用了一种相当明显和直接的方法,类似于本主题中提到的方法。 HTML 5 video or audio playlist
这种方法的问题在于,每当一个视频结束并且后续视频被指定为新源时,仍然需要加载新视频,从而导致可能存在长时间差距。如果甚至可以在video.src中命名之前强制所有视频都进行预加载,那么这种方法仍然有效。
尝试#2 我重写了脚本,因此列表中的每个视频都会导致创建一个单独的视频元素,但不会附加到文档中(使用createElement)。然后,当每个视频结束时,我删除了前一个节点并附加了下一个元素。
在页面加载时执行的代码:
for (i in timelinePlay) if (timelinePlay[i] != null) { var element = document.createElement('video'); element.id = 'video1-' + (i); element.src = timelinePlay[i]['URL']; element.preload = 'load'; video1Array[i] = element; }
视频'已结束'事件代码:
videoElement.addEventListener("ended", function (event) { if (this.currentTime > 0) { if (player.hasChildNodes()) while (player.childNodes.length > 0) player.removeChild(player.firstChild); player = document.getElementById('canvas-player'); player.appendChild(video1Array[timelineCurrent]); nextVideo = document.getElementById('video1-' + timelineCurrent); nextVideo.play(); timelineCurrent++; } }, false);
(请注意,这些代码示例是部分的,并且有些简化。另外,我确实认识到将对象用作关联数组不是最佳实践)
这次修改的结果更接近,因为视频现在在他们需要播放时加载,但视频之间仍然存在短暂且不一致的差距。
尝试#3 我用'timeupdate'监听器替换了'ends'事件监听器,开头如下:
nextVideo.addEventListener("timeupdate", function (event) { if (this.currentTime > (this.duration - 0.1) && this.currentTime > 1) { this.removeEventListener("timeupdate", function () { return; }, false);
('this.currentTime> 1'只是为了确保以前的视频实际播放)
我希望视频之间的差距足够接近以保持一致,即在上一个视频结束之前的任意0.1秒开始下一个视频会将间隙纠正到可接受的程度。结果是时间确实更好,但它正在跳过视频。我将跳过的视频归因于'timeupdate'事件的失误,但我可能会弄错。
我也考虑过的其他替代方案: SMIL脚本(似乎已过时,并且可能会有相同的同步问题) ffmpeg在后端连接视频(我的主机不允许shell脚本)
仅供参考我目前正在为Safari 5.0.3开发
答案 0 :(得分:3)
由于你的提示,我有一个类似的问题,我现在设法解决了。结果是一个可能动态的视频元素列表,可以一个接一个地播放而没有间隙。
我在几个视频元素上使用jwplayer而不是原生视频标签。 在启动时,所有元素开始播放一秒,暂停并重绕为零。 然后逐个播放,并通过display:block。
显示<ul class="playlist">
<li>
<video class="playlist" id="vid01" poster="img/preview.jpg">
<source src="vid/v01.mp4" type="video/mp4">
</video>
</li>
<li>
<video class="playlist" id="vid02" poster="img/preview.jpg">
<source src="vid/v02.mp4" type="video/mp4">
</video>
</li>
<li>
<video class="playlist" id="vid03" poster="img/preview.jpg">
<source src="vid/v03.mp4" type="video/mp4">
</video>
</li>
<li>
<video class="playlist" id="vid04" poster="img/preview.jpg">
<source src="vid/v04.mp4" type="video/mp4">
</video>
</li>
</ul>
On添加到舞台上,“预加载”视频一两秒钟: (摘录)
var isPrebuffering = false,
isSeeking = false;
$player = jwplayer(videoId).setup({
width: '800px',
height: '450px',
autoplay: false,
modes: [
{ type: 'html5' }
],
events: {
onReady: function(e) {
prebuffer();
},
// There is a quirk in jwplayer that makes it impossible to pause at the
// new position directly after seeking. So we have to work around that.
onTime: function(e) {
if((e.position > 1.0 && isPrebuffering) || isSeeking) {
finishPrebuffering();
}
}
}
});
function prebuffer() {
isPrebuffering = true;
$player.setMute(true);
$player.play();
};
function finishPrebuffering() {
if(isPrebuffering) {
isPrebuffering = false;
isSeeking = true;
$player.seek(0);
}
if($player.getPosition() === 0) {
isSeeking = false;
$player.pause();
$player.setMute(false);
}
};
答案 1 :(得分:0)
HTML5的mediaElement接口有时会在preload,load和buffered之间相当混乱。 但是,您是否尝试在视频结束前3秒检测视频事件? 然后在ajax中尝试在新的div中加载下一个视频。 在您的视频结束事件中,您可以切换div,以便您的下一个视频足够加载以进行播放。 等等下一个视频。
答案 2 :(得分:0)
您可以使用http://popcornjs.org/执行此操作。他们使用html5视频标签在一帧一帧的视频中提供js钩子。有很多事件等。