- xml - AJAX/Jquery XML 解析
- 具有多重继承的 XML 模式
- .net - 枚举序列化 Json 与 XML
- XML 简单类型、简单内容、复杂类型、复杂内容
编辑 - 2016-06-25
(我删除了 6 月 16 日的更新,因为不再相关。
我把我的 OP 留在了下面...)
我今天又花了 4 个小时在这上面。
现在是这样的情况:
我的函数 WORKS 用于两个链接。
视频参数在数组中定义。
其他两个链接无效。
我得到一个 Uncaught TypeError: thisPlayer.loadVideoById is not a function
对于链接 #3 和 #4。
但是相同的功能适用于链接 #1 和 #2。
似乎 youtube 对象仅针对两个第一定义。
为什么?
请仔细查看此实时链接上的控制台:
https://www.bessetteweb.com/?p=youtube-video-test
我插入了很多 console.log 消息以使其清晰。
这是我的实际代码:
// Global variable for the player
var player = [];
var statePlaying=false;
playerArr = [{
linkID:"link0",
divID:"player1",
ytID:"5V_wKuw2mvI", // Heavy metal playlist
start:20,
end:30
},
{
linkID:"link1",
divID:"player2",
ytID:"u9Dg-g7t2l4", // Disturbed
start:10,
end:20
},
{
linkID:"link2",
divID:"player3",
ytID:"39b5v3-d6ZA", // Maiden
start:30,
end:40
},
{
linkID:"link3",
divID:"player4",
ytID:"z8ZqFlw6hYg", // Slayer
start:120,
end:136
}];
// This function gets called when API is ready to use
function onYouTubePlayerAPIReady() {
// Binding events loop
console.log("playerArr.length: "+playerArr.length);
for(i=0;i<playerArr.length;i++){
console.log("");
console.log("onPlayerReady for loop ->i: "+i);
var playButton = document.getElementById(playerArr[i].linkID);
console.log("playButton.id: "+playButton.id);
var thisArr = playerArr[i];
console.log("playerArr[i] object (below): ");
console.log(thisArr);
playButton.addEventListener("click", function() {
thisLinkID = $(this).attr("id").replace("link","");
console.log("");
console.log("------------------");
console.log("Link #"+(parseInt(thisLinkID)+1)+" clicked.");
console.log("------------------");
var ytID = playerArr[thisLinkID].ytID;
var start = playerArr[thisLinkID].start;
var end = playerArr[thisLinkID].end;
var thisPlayer = new YT.Player(playerArr[thisLinkID].divID);
console.log("ytID: "+ytID);
console.log("start: "+start);
console.log("end: "+end);
console.log("");
console.log("Below are the google ads, blocked by AdBlocker.");
$("#ytplayerModal").css({"display":"block"});
$("#ytplayerModal").animate({"opacity":"0.7"},1000,function(){
console.log("");
console.log("player show()");
$(".ytplayer").show();
console.log("Youtube player object:");
console.log(thisPlayer);
console.log("");
console.log('loadVideoById() parameters --\> videoId:'+ytID+', startSeconds:'+start+', endSeconds:'+end);
thisPlayer.loadVideoById({'videoId':ytID, 'startSeconds':start, 'endSeconds':end});
console.log("");
});
// Bugfix - Set Interval instead of listener
setTimeout(function(){
var IntervalCounter=0;
listenerInterval = setInterval( function() {
var state = thisPlayer.getPlayerState(); //player[thisLinkID].getPlayerState();
var stateMsg;
switch (state){
case -1: stateMsg="unstarted"; thisPlayer.playVideo(); console.log("player["+thisLinkID+"]"); break;
case 0: stateMsg="ended"; break;
case 1: stateMsg="playing"; break;
case 2: stateMsg="paused"; break;
case 3: stateMsg="buffering"; break;
case 5: stateMsg="video cued"; break;
default: stateMsg="Undefined player state...";
}
console.log(state+" : "+stateMsg);
if(state==1){
statePlaying=true;
}
// Closes the modal
if((statePlaying) && (state==0)){
//if((statePlaying) && (stateObj.data==0)){
setTimeout(function(){
console.log("Closing Modal");
$(".ytplayer").css({"display":"none"});
$("#ytplayerModal").animate({"opacity":"0"},1000,function(){
$("#ytplayerModal").css({"display":"none"});
});
statePlaying=false;
},500);
clearInterval(listenerInterval);
console.log("Interval loop stopped on video end.")
}
// Stop the interval at 1000... Endless instead!
IntervalCounter++;
if((IntervalCounter>999)&&(state!=1)&&(state!=2)&&(state!=0)){
clearInterval(listenerInterval);
console.log("Interval loop willingly stopped. Endless otherwise.")
}
}, 10);
},1100); // Interval setTimeout
});
}
}
// Inject YouTube API script
var tag = document.createElement('script');
tag.src = "//www.youtube.com/player_api";
var firstScriptTag = document.getElementsByTagName('script')[0];
firstScriptTag.parentNode.insertBefore(tag, firstScriptTag);
----------------
这是我的原始帖子 - 6 月 14 日:
我已经在这上面花了 6 个小时。
(加上 1 个编辑这个问题!)
我正在调试它的最后 4 行。
问题是我没有收到控制台错误提示。
概念:
我想将大量 Youtube 视频链接到文本链接。
用户想要的效果是能够在阅读文本时单击引文链接...以便能够确认引文。
视频不必完全播放。
视频应以特定时间码开始并以特定时间码结束。
额外的复杂性:所有这些都希望以模态视图样式显示。
我的代码工作对于ONE 视频来说相当快。 <强> See here 。
我的代码基于 this tutorial并且很快就成功了。
然后,让这个工作...
我需要构建数组来处理多个视频。
对于链接 ID、播放器 ID、开始/结束的时间码……以及听众!
乐趣开始了!
如前所述,我一天的大部分时间都花在了这上面。我总是遇到带有控制台错误的错误作为清晰的(LOLLL)指南。
我对自己的工作很满意......我认为这是朝着正确的方向前进。
这几乎可以工作了......
但这一次,没有错误! See here . (检查控制台!)
什么!!!没有错误?!?
我的 ARM 现在被砍掉了。
事实上,第一个显示但视频没有开始......第二个看起来完全迷失在阴霾中。
在控制台日志消息中,我看到第一次出现 onStateChange
监听器,即 -1(未启动)。但是之后 ???它死了!
啊!
我必须超越我的骄傲……并将它作为一个问题带到 StackOveflow 上。
;)
我的完整代码(用于多个链接):
这是一个通过 ajax 调用的页面...因此所有外部资源(如 jQuery)都已加载。
<style>
.ytplayer{
position:fixed;
z-index:2;
width:60%;
height:40%;
top:30%;
left:20%;
display:none;
}
#ytplayerModal{
display:none;
background-color:#000;
opacity:0;
position:fixed;
z-index:1;
top:0;
left:0;
width:100%;
height:100%;
}
.ytTriggerPlay{
text-decoration:underline;
color:dodgerblue;
cursor:pointer;
}
</style>
<h1>Youtube modal trigger link test</h1>
<br>
<br>
<div id="text">
Lorem ipsum dolor sit amet, consectetur <a id="0" class="ytTriggerPlay">adipiscing elit</a>. Quisque feugiat lectus ut est vestibulum ornare. Vivamus felis nulla, facilisis id cursus non, pharetra non diam. Sed pellentesque turpis vel sem tincidunt consectetur. Aenean ut lorem erat. Donec ut tellus sed leo ultrices cursus. <a id="1" class="ytTriggerPlay">Cras varius libero</a> ut purus suscipit ultrices. Vivamus eget efficitur turpis. Aenean suscipit, dui nec luctus fringilla, neque tellus fringilla risus, et porta enim justo et turpis. Sed risus orci, vehicula sed eleifend eget, tincidunt ut turpis. Vestibulum in sapien non lacus tristique mattis id eget tortor.<br>
<br>
Proin est purus, maximus id nunc vel, consectetur tristique urna. Mauris cursus ipsum a varius luctus. Nunc interdum condimentum massa vitae rutrum. Morbi volutpat nec lorem eleifend malesuada. Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas. Duis fringilla metus vel nunc elementum efficitur. Duis sed dolor diam. In eu ultrices libero, eget lobortis mi. Sed pretium orci non augue vehicula, eget placerat leo lacinia. Sed sed gravida dui. Lorem ipsum dolor sit amet, consectetur adipiscing elit. In bibendum, erat eget venenatis elementum, nulla enim posuere lacus, quis efficitur dolor ex quis ipsum. Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas. Phasellus volutpat finibus odio id venenatis. Fusce at leo libero. Cras eget velit sed justo egestas vehicula efficitur sit amet ex.<br>
</div>
<!--iframe id="ytplayer" type="text/html" width="720" height="405" src="https://www.youtube.com/embed/5V_wKuw2mvI?end=60&start=20" frameborder="0" allowfullscreen-->
<div id="ytplayerModal"></div>
<div id="player1" class="ytplayer"></div>
<div id="player2" class="ytplayer"></div>
<script>
// https://developers.google.com/youtube/iframe_api_reference
// https://css-tricks.com/play-button-youtube-and-vimeo-api/
// Global variable for the player
var player = [];
var statePlaying=false;
playerArr = [{
linkID:"0",
divID:"player1",
ytID:"5V_wKuw2mvI", // Heavy metal playlist
start:20,
end:40,
},
{
linkID:"1",
divID:"player2",
ytID:"39b5v3-d6ZA", // Maiden
start:30,
end:60,
}];
// This function gets called when API is ready to use
function onYouTubePlayerAPIReady() {
for(i=0;i<playerArr.length;i++){
// Create the global player from the specific iframe (#video)
thisPlayer = new YT.Player(playerArr[i].divID, {
height: '352',
width: '640',
videoId: '5V_wKuw2mvI',
startSeconds:20,
endSeconds:40,
events: {
// Call this function when player is ready to use
// 'onReady': onPlayerReady // Commented out willingly.
}
});
player[i] = thisPlayer;
}
onPlayerReady();
}
function onPlayerReady(event) {
// Binding events loop
console.log("playerArr.length: "+playerArr.length);
for(i=0;i<playerArr.length;i++){
console.log("");
console.log("onPlayerReady for loop ->i: "+i);
var playButton = document.getElementById(playerArr[i].linkID);
console.log("playButton.id: "+playButton.id);
var thisArr = playerArr[i];
console.log("playerArr[i] object (below): ");
console.log(thisArr);
var thissPlayer = player[i];
playButton.addEventListener("click", function() {
thisLinkID = parseInt($(this).attr("id"));
console.log("thisLinkID: "+thisLinkID);
var ytID = playerArr[thisLinkID].ytID;
var start = playerArr[thisLinkID].start;
var end = playerArr[thisLinkID].end;
console.log("ytID: "+ytID);
console.log("start: "+start);
console.log("end: "+end);
console.log("thissPlayer object (below): ");
console.log(thissPlayer);
$("#ytplayerModal").css({"display":"block"});
$("#ytplayerModal").animate({"opacity":"0.7"},1000,function(thissPlayer,ytID,start,end){
$(".ytplayer").show();
player[thisLinkID].loadVideoById({videoId:ytID, startSeconds:start, endSeconds:end});
setTimeout(function(){
player[thisLinkID].playVideo();
},500);
});
});
thissPlayer.addEventListener("onStateChange", function(stateObj){
console.log("Player State: "+stateObj.data);
console.log("Again, thissPlayer object in the onStateChange listener (below).");
console.log(thissPlayer);
// State sequence : -1, 3, 1, 2, 0, which is: Unstarted, Buffering, Playing, Paused, Ended.
if(stateObj.data==1){
statePlaying=true;
}
console.log("Player State bolean memory: "+statePlaying);
// Closes the modal
if((statePlaying) && (stateObj.data==0)){
setTimeout(function(){
console.log("Closing Modal");
$(".ytplayer").css({"display":"none"});
$("#ytplayerModal").animate({"opacity":"0"},1000,function(){
$("#ytplayerModal").css({"display":"none"});
});
statePlaying=false;
},500);
}
});
}
}
// Inject YouTube API script
var tag = document.createElement('script');
tag.src = "//www.youtube.com/player_api";
var firstScriptTag = document.getElementsByTagName('script')[0];
firstScriptTag.parentNode.insertBefore(tag, firstScriptTag);
</script>
最佳答案
iFrame 播放器 API 存在一个临时问题,您可以在此处阅读:https://code.google.com/p/gdata-issues/issues/detail?id=4706
作为临时修复,您只需要在 onReady 事件中添加事件监听器:
function onReady() {
player.addEventListener('onStateChange', function(e) {
console.log('State is:', e.data);
});
}
此外,正如有人在 Google 代码问题线程中提到的那样,您可以设置一个时间间隔并轮询播放器的当前状态,而不是监听 onStateChange 事件。这是执行此操作的示例代码片段:
setInterval( function() {
var state = player.getPlayerState();
if ( playerState !== state ) {
onPlayerStateChange( {
data: state
});
}
}, 10);
关于Javascript 事件监听器退出(?)监听... Youtube API - 无控制台错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37823490/
当单击复选框(或选择所有复选框)时,我想向 CheckboxSelectionModel 添加一个监听器。 var smSensors = new xg.CheckboxSelectionModel(
我有一个简单的程序,允许 2 个客户端连接到服务器。 连接后,他们可以轮流点击空白卡片图像。 一旦 2 个客户中的任何一个点击空白卡片图片,卡片图片将变为 Ace 俱乐部图片。 更改将显示在客户端的两
我在这里看到了一个代码,该代码以字符串的形式检索鼠标的当前图标,但是此代码使用了TTimer来实现。 因此,我想知道是否存在某些事件(侦听器)来检测鼠标光标图标上的这些更改。 下面是使用TTimer的
我想在我的配置对象上获得一个 onload 事件。 以下工作,除非我创建一个 config.listeners={..} (我认为这就是我需要的?)替换 this.onload({...}); 我什至
通常,在 Java 中,当我有一个向其他对象提供某种通知的对象时,我将使用 Listener/Observer 模式。 有没有更类似于 Scala 的方式来做到这一点?我应该在 Scala 中使用这种
我有一个带有动画器的游戏对象和一些可以触发事件的动画(具有特定的回调函数)。 只要我将脚本添加到与动画器相同的游戏对象(包括接收器),一切都会正常工作: public class AnimatorEv
我有一个带有监听器的 DialogFragment,用于单击按钮以调用 fragment 中的函数。 我收到 lateinit property listener has not been initi
这个问题已经有答案了: Java ActionListener error: incompatible types (4 个回答) 已关闭 5 年前。 我最近刚刚开始学习 Java 代码。我在添加监听
我的代码遇到问题。我想知道是否有一种更简单的方法来使用监听器,而不是不断地这样做: example.addActionListener(new java.awt.event.ActionListene
有没有办法使用 .net 创建控制台应用程序。或通过某个端口监听 SMTP 消息的服务? 我需要创建一个中间层对象来捕获和处理 smtp 消息。也就是说,我希望该监听器发送和接收 smtp 消息,然后
我有一个带有动画器的游戏对象和一些可以触发事件的动画(具有特定的回调函数)。 只要我将脚本添加到与动画器相同的游戏对象(包括接收器),一切都会正常工作: public class AnimatorEv
我有许多向主事件生成服务注册的监听器。然而,我想告诉听众,事件流在某个时刻将会结束。您会通过简单地调用监听器上的方法(例如 finish())来完成此操作,还是有一个单独的事件方法 streamFin
我的代码有什么问题。 我创建了一个 JList,添加了项目并将其推到左侧(BorderLayout.WEST)。每次单击列表项时,我希望在列表右侧显示一个面板。但问题是,当选择列表项并运行监听器时,到
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 8 年前。 Improve th
这可能是一个简单的问题,但我没有看到它。 我有一个界面 public interface RenderableListener{ public void update(T element);
有人可以直接指出我的正确方向吗?当从组合框中选择适当的选项时,我希望小程序中的 Action 监听器显示从 html 文件检索的 jlabel 中的 3 个参数之一。 干杯 最佳答案 对于组合框,您需
我有一个网站,每个页面上都有许多 jQuery 事件处理程序,所有这些都在一个大型 .js 文件中。 这意味着对于任何页面,大多数事件处理程序都是针对不存在且根本不会使用的 HTML。 这会影响我的表
我有一些 jQuery 监听器设置,用于监听 type="text" 字段上的表单输入。但是,当用户从自动完成下拉框中选择一个选项(即他们之前输入的值已被记住以供将来使用)时,下面的监听器不会收集该值
关闭。这个问题需要多问focused 。目前不接受答案。 想要改进此问题吗?更新问题,使其仅关注一个问题 editing this post . 已关闭 9 年前。 Improve this ques
我正在用 unity (c#) 做一个类似国际象棋的游戏,但我在尝试进行向上转换以将信息从一个 child 发送到另一个抽象类时遇到了困难。 基本上,我有一个抽象类,它有一个带有函数的事件/委托(de
我是一名优秀的程序员,十分优秀!