如何解决A帧切换模板导致声音视频?重叠 我的行为已修改:我的猜测:示例:如何重现我的问题:示例 2一个“有点”的工作示例
我的行为(已修改):
我正在使用 A-Frame 和 A-Frame 模板组件。
我使用模板组件从 script id="..."> 标签加载模板,例如 example。
我想使用 el.setAttribute 动态更改模板,就像在这个 example 中一样。
我的模板包含一个以视频为源的天空>元素。
我可以加载一个模板,它就像一个魅力。我可以暂停、播放等等。
当我想将模板切换到使用相同视频资源的新模板,或者加载一个新模板然后返回到第一个模板时,会出现此问题。
显示一个视频,但在后台播放第二个声音。我还可以验证声音不是来自显示的视频。我仍然可以播放/暂停它,它的声音会在后台播放。
已编辑
我不仅希望能够修改 sky> 元素,还希望能够修改其他一些元素,这就是我想使用模板组件的原因。假设我不知道最终会使用哪些资产,并且我不想使用资产管理系统来加载数十 Mb 的视频/对象等并在加载屏幕上暂停用户。如here所述:
在浏览器获取(或出错)所有资产或资产系统达到超时之前,场景不会渲染或初始化。
为了进一步澄清我的意图,但仍然保持简单,我希望用户能够通过选择预制件来改变场景(假设每个视频 15Mb,即使有 6-7 个资产,它也可能几乎达到 100Mb),或通过上传一个。而且我真的很想避免重新加载页面,因为从用户的角度来看,这不是一个优雅的解决方案。
我的猜测:
我认为旧视频并没有完全处理掉。我尝试在各种元素/材料/视频本身上使用three.js .dispose() 方法,但没有成功。
我还尝试使用 a-video> 作为源,并在切换前删除元素,但结果相同。
问题
示例:
function switchVideoToFirst() {
console.log("switching to first");
document.getElementById("skyVideoTemplate").setAttribute('template','src',"#firstVideo");
}
function switchVideoToSecond() {
console.log('switching to second');
document.getElementById("skyVideoTemplate").setAttribute('template','#secondVideo');
}
function playVideo() {
document.getElementById("skyVideo").components.material.material.map.image.play();
}
function pauseVideo() {
document.getElementById("skyVideo").components.material.material.map.image.pause();
}
<!DOCTYPE html>
<head>
<Meta charset="UTF-8">
<script src="https://cdnjs.cloudflare.com/ajax/libs/nunjucks/2.3.0/nunjucks.min.js"></script>
<script src="https://aframe.io/releases/1.1.0/aframe.min.js"></script>
<script src="https://unpkg.com/aframe-template-component@3.2.1/dist/aframe-template-component.min.js"></script>
</head>
<body>
<div style="width: 400px; height: 200px;">
<a-scene embedded>
<a-entity id="skyVideoTemplate" template="src: #"></a-entity>
</a-scene>
</div>
<button onclick="switchVideoToFirst()">switch video to first</button>
<button onclick="switchVideoToSecond()">switch video to second</button>
</br>
</br>
<button onclick="playVideo()">play</button>
<button onclick="pauseVideo()">Pause</button>
</body>
<script id="firstVideo" type='text/nunjucks'>
<a-sky id="skyVideo" src="https://upload.wikimedia.org/wikipedia/commons/transcoded/f/fb/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm.360p.vp9.webm">
</a-sky>
<a-Box position="-1 1 -1" color="red"></a-Box>
</script>
<script id="secondVideo" type='text/nunjucks'>
<a-sky id="skyVideo" src="https://upload.wikimedia.org/wikipedia/commons/transcoded/f/fb/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm.360p.vp9.webm">
</a-sky>
<a-sphere position="-1 1 -1" color="green"></a-sphere>
</script>
我知道我在两个模板中都有相同的视频,但这是有意缩短示例。该问题与重新加载相同的模板密切相关,要么切换到不同的模板,要么加载具有相同视频资源的新模板。
如何重现我的问题:
- 按“将视频切换到第一个”以加载第一个模板。
- 按“将视频切换到第二个”以加载第二个模板。
- 视频消失了,但声音还在。
- 这点可能不起作用,但如果您等待的时间足够长,第二个视频将加载并且您会听到 2 种声音。然后播放/暂停控件将只对其中之一起作用。
我也经历过:
- 按“将视频切换到第一个”加载第一个模板
- 暂停视频
- 按“切换视频到第二个”以加载第二个模板
- 白屏可能需要一段时间
- 加载“第二个”视频时,“第一个”视频仍会显示并暂停,但新视频会从头开始播放声音。那么您应该(或不知道,无法弄清楚)能够播放/暂停“第一个”视频。
示例 2
在 Piotr Adam Milewski 的大力投入之后,我尝试了以下操作
更改模板时的声音消失了,但加载第二个模板后声音从头开始但视频暂停。
AFRAME.registerComponent("pausewhenremoved",{
init: function() {
this.el.addEventListener('materialvideoloadeddata',evt => {
this.videoElement = evt.detail.src;
//this.videoElement.pause();
});
},remove() {
// once i'm removed,the video is going down too
this.videoElement.pause();
// I also tried removing the video from DOM but this only changed video to blank (sound was still on)
//this.videoElement.removeAttribute('src');
//this.videoElement.load();
//$(this.videoElement).empty().remove();
}
});
function switchVideoToFirst() {
console.log("switching to first");
document.getElementById("skyVideoTemplate").setAttribute('template','#secondVideo');
}
<!DOCTYPE html>
<head>
<Meta charset="UTF-8">
<script src="https://cdnjs.cloudflare.com/ajax/libs/nunjucks/2.3.0/nunjucks.min.js"></script>
<script src="https://aframe.io/releases/1.1.0/aframe.min.js"></script>
<script src="https://unpkg.com/aframe-template-component@3.2.1/dist/aframe-template-component.min.js"></script>
</head>
<body>
<div style="width: 400px; height: 200px;">
<a-scene embedded>
<a-entity id="skyVideoTemplate" template="src: #"></a-entity>
</a-scene>
</div>
<button onclick="switchVideoToFirst()">switch video to first</button>
<button onclick="switchVideoToSecond()">switch video to second</button>
</body>
<script id="firstVideo" type='text/nunjucks'>
<a-sky id="skyVideo" pausewhenremoved src="https://upload.wikimedia.org/wikipedia/commons/transcoded/f/fb/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm.360p.vp9.webm">
</a-sky>
<a-Box position="-1 1 -1" color="red"></a-Box>
</script>
<script id="secondVideo" type='text/nunjucks'>
<a-sky id="skyVideo" pausewhenremoved src="https://upload.wikimedia.org/wikipedia/commons/transcoded/f/fb/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm.360p.vp9.webm">
</a-sky>
<a-sphere position="-1 1 -1" color="green"></a-sphere>
</script>
一个“有点”的工作示例
我创建了一个标准的 HTML 视频元素来存储视频并修改其源,而不是将其留给 A-Frame。
您可以在两个模板之间切换,只要不要切换得太快,它就会正常渲染(如果您尝试切换太早,屏幕会变黑但声音正常)。
>function switchVideoToFirst(buttonElement) {
console.log("switching to first");
if(document.getElementById("sky")) {
document.getElementById("sky").remove();
}
document.getElementById("skyVideoTemplate").setAttribute('template',"#firstVideo");
}
function switchVideoToSecond(buttonElement) {
console.log('switching to second');
if(document.getElementById("sky")) {
document.getElementById("sky").remove();
}
document.getElementById("skyVideoTemplate").setAttribute('template','#secondVideo');
}
<!DOCTYPE html>
<head>
<Meta charset="UTF-8">
<script src="https://cdnjs.cloudflare.com/ajax/libs/nunjucks/2.3.0/nunjucks.min.js"></script>
<script src="https://aframe.io/releases/1.1.0/aframe.min.js"></script>
<script src="https://unpkg.com/aframe-template-component@3.2.1/dist/aframe-template-component.min.js"></script>
</head>
<body>
<script>
AFRAME.registerComponent("pausewhenremoved",{
schema: {
src: {
type: 'string',default: ''
}
},init: function() {
this.videoElement = document.getElementById("skyVideoHolder");
this.videoElement.setAttribute('src',this.data.src);
this.videoElement.load();
},play() {
if (this.videoElement) {
this.videoElement.play();
}
},pause() {
if (this.videoElement) {
this.videoElement.pause();
}
},the video is going down too
this.videoElement.pause();
}
});
</script>
<div style="width: 400px; height: 200px;">
<a-scene embedded>
<video id="skyVideoHolder" src="" loop></video>
<a-entity id="skyVideoTemplate" template="src: #"></a-entity>
</a-scene>
</div>
<button onclick="switchVideoToFirst(this)">switch video to first</button>
<button onclick="switchVideoToSecond(this)">switch video to second</button>
</body>
<script id="firstVideo" type='text/nunjucks'>
<a-sky id="skyVideo" pausewhenremoved="src: https://upload.wikimedia.org/wikipedia/commons/transcoded/f/fb/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm.360p.vp9.webm" src="#skyVideoHolder">
</a-sky>
<a-Box position="-1 1 -1" color="red"></a-Box>
</script>
<script id="secondVideo" type='text/nunjucks'>
<a-sky id="skyVideo" pausewhenremoved="src: https://upload.wikimedia.org/wikipedia/commons/transcoded/f/fb/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm.360p.vp9.webm" src="#skyVideoHolder">
</a-sky>
<a-sphere position="-1 1 -1" color="green"></a-sphere>
</script>
免责声明
示例中使用的 360 度全景视频来自 this website,作者是 Jan Ainali
解决方法
我认为您直接遇到了 this 问题 - 一旦 a-frame
从内联 URL 创建 <video>
它有点忘记它。这是相关的,因为模板组件只是添加和 removing 元素。
所以问题的一个很大的简化是 - 被移除的 videosphere 仍然在播放视频:
setTimeout(e => {
let sky = document.querySelector("a-sky")
sky.parentNode.removeChild(sky)
},2000)
<script src="https://aframe.io/releases/1.2.0/aframe.min.js"></script>
<a-scene background="color: #ECECEC">
<a-sky id="skyVideo" src="https://upload.wikimedia.org/wikipedia/commons/transcoded/f/fb/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm.360p.vp9.webm">
</a-sky>
</a-scene>
解决这个问题的一种方法是创建一个组件,它将抓取加载的视频元素,并在实体移除时暂停/移除它:
setTimeout(e => {
let sky = document.querySelector("a-sky")
sky.parentNode.removeChild(sky)
},2000)
<script src="https://aframe.io/releases/1.2.0/aframe.min.js"></script>
<script>
AFRAME.registerComponent("foo",{
init: function() {
// once the video data is loaded,store the element
this.el.addEventListener("materialvideoloadeddata",evt => {
this.videoEl = evt.detail.src;
})
},remove() {
// once i'm removed,the video is going down too
this.videoEl.pause();
}
})
</script>
<a-scene background="color: #ECECEC">
<a-sky foo id="skyVideo" src="https://upload.wikimedia.org/wikipedia/commons/transcoded/f/fb/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm.360p.vp9.webm">
</a-sky>
</a-scene>
另一种方法是使用资产管理系统 - 您可以在其中控制视频 HTML 元素:
let video = document.querySelector("#vid");
let sky = document.querySelector("a-sky");
video.addEventListener("play",e => {
setTimeout(e => {
video.pause();
sky.parentNode.removeChild(sky);
},4000);
});
<script src="https://aframe.io/releases/1.2.0/aframe.min.js"></script>
<a-scene background="color: #ECECEC">
<a-assets>
<video id="vid" autoplay loop="true" preload="auto" crossorigin="anonymous" src="https://upload.wikimedia.org/wikipedia/commons/transcoded/f/fb/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm/Hundra_knektars_marsch_p%C3%A5_Forum_Vulgaris.webm.360p.vp9.webm"></video>
</a-assets>
<a-sky id="skyVideo" src="#vid"> </a-sky>
</a-scene>
另一方面,如果您想更改/重新启动视频,并将立方体切换到球体,我不会使用 template
组件,只需管理视频 HTML 元素,然后更改立方体 /球体可见性。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。