// ==UserScript==
// @name 油管字幕放到视频下方
// @namespace 0x90DD
// @version 0.0.2
// @description 油管字幕在视频上,看起来不舒服,挪到视频下方了。字幕插件地址 https://chrome.google.com/webstore/detail/language-reactor/hoombieeljmmljlkjmnheibnpciblicm
// @author 0x90DD
// @match https://www.youtube.com/*
// @match https://www.youtube.com/watch*
// @license GPL-3.0 License
// @run-at document-start
// @icon 
// @require https://cdn.bootcdn.net/ajax/libs/jquery/3.5.1/jquery.min.js
// @grant unsafeWindow
// @grant GM_registerMenuCommand
// @grant GM_setValue
// @grant GM_getValue
// @run-at context-menu
// ==/UserScript==
(function () {
'use strict';
// Your code here...
let changed = false;
let outerHtml;
let root;
let bottomParent;
let caption;
let uncle;
let videoCationsShell;
let lastLlnVisibility = undefined;
let isInTheater = undefined;
let notificationParent;
let restHeight;
GM_registerMenuCommand("点击单词时打开查词窗口 打开/关闭", onSwitchOpenBing, "H");
function onSwitchOpenBing() {
let v = GM_getValue("openBing");
v = !!!v;
GM_setValue("openBing", v);
alert(`点击单词时直接打开翻译窗口已${v ? "启用" : "停用"}`);
}
later();
function later() {
setTimeout(() => {
checkCaption();
}, 100 * 3);
}
GM_registerMenuCommand("隐藏中间字幕 打开/关闭", onSwitchHideLlnBottom, "H");
function onSwitchHideLlnBottom() {
let v = GM_getValue("openLB");
v = !!!v;
GM_setValue("openLB", v);
const adj = document.querySelector(".lln-bottom-panel ");
if (!v)
{
adj.setAttribute('style', 'display:hidden');
}
else{
adj.setAttribute('style', 'display:visible;position:static;background-color:#000000;height:16rem;border:1px solid #6f6f6f');
}
alert(`隐藏中间字幕已${v ? "启用" : "停用"}`);
}
function laterCheckVideo() {
setTimeout(laterCheckVideo, 1000);
let notification = document.querySelector(".lln-notification");
if (notification && notification.parentNode != notificationParent) {
notificationParent = document.createElement("div");
notificationParent.style.cssText = 'position: fixed;left:-10000px;display:none;';
let body = document.querySelector("body");
body.appendChild(notificationParent);
notificationParent.appendChild(notification);
console.log("notifactionParent is", notificationParent);
}
if (document.fullscreen) {
return;
}
let v = document.querySelector("video");
if (!v || v.videoWidth == 0) {
return;
}
v.setAttribute('style', 'position:static');
let top = v.style.top;
let t = document.querySelector("#tttt");
if (!t) {
t = document.createElement("div");
videoCationsShell = t;
t.setAttribute('id', 'tttt');
v.parentNode.appendChild(t);
}
let c = document.getElementsByClassName("lln-vertical-view lln-events-added");
if (!c || c.length == 0) return;
let d = c[0];
let visibility = window.getComputedStyle(d, null).getPropertyValue("visibility");
let playerParent = document.querySelector("#player-theater-container")
if (!playerParent) return;
let isTheater = playerParent.childNodes.length > 2;
let restHeight = window.getComputedStyle(d).getPropertyValue('height').replace('px', '') - window.getComputedStyle(v, null).getPropertyValue("height").replace('px', '');
checkChangeParent(visibility, isTheater, restHeight, v);
}
function checkChangeParent(visibility, isTheater, height, v) {
if (visibility == lastLlnVisibility && isTheater == isInTheater && restHeight == height) {
return
}
if (!bottomParent) {
return;
}
if (!videoCationsShell) {
return;
}
if (document.fullscreen) {
return;
}
lastLlnVisibility = visibility;
isInTheater = isTheater;
restHeight = height;
if (lastLlnVisibility == "visible" && isInTheater && restHeight > 160) {
if (caption.parentNode == bottomParent) {
bottomParent.removeChild(caption);
}
if (caption.parentNode != videoCationsShell) {
videoCationsShell.appendChild(caption);
}
} else {
if (caption.parentNode == videoCationsShell) {
videoCationsShell.removeChild(caption);
}
if (caption.parentNode != bottomParent) {
bottomParent.insertBefore(caption, uncle.nextSibling);
}
}
}
laterCheckVideo();
function checkCaption() {
if (changed) return;
root = document.querySelector("#primary-inner");
if (!root) return later();
var player = document.querySelector("#primary-inner").querySelector("#player");
uncle = player
if (!uncle) return later();
caption = document.querySelector("#lln-bottom-panel");
if (!caption) return later();
if (!outerHtml) {
var br = document.createElement('br');
br.setAttribute(
'style',
'background-color:"#FFFFFF";height:"1px";width:"100%";',
)
outerHtml = caption.interHTML;
}
const content = document.querySelector(".lln-youtube .lln-bottom-panel .lln-subs-wrap");
if (!content) return later();
const sub = document.querySelector(".lln-bottom-panel #lln-main-subs");
if (!sub) return later();
const subWrap = document.querySelector(".lln-bottom-panel .lln-subs-wrap");
if (!subWrap) return later();
const youtubebottommain = document.querySelector(".lln-youtube .lln-bottom-panel #lln-main-subs");
if (!youtubebottommain) return later();
const adj = document.querySelector(".lln-bottom-panel .lln-subs-font-adjust");
if (!adj) return later();
adj.setAttribute('style', 'padding-top:-15px');
youtubebottommain.setAttribute('style', 'justify-content:flex-start');
sub.setAttribute('style', 'justify-content:flex-start');
document.querySelector("#lln-sub-view-wrap").parentElement.removeChild(document.querySelector("#lln-sub-view-wrap"));
subWrap.setAttribute('style', 'padding-left: 2px;padding-right: 2px');
content.setAttribute('style', 'padding-top:1px');
caption.setAttribute('style', 'position:static;background-color:#000000;height:16rem;border:1px solid #6f6f6f');
bottomParent = uncle.parentNode;
bottomParent.insertBefore(caption, uncle.nextSibling);
changed = true;
lastLlnVisibility = undefined;
}
function toFull() {
var n = document.createElement("div");
n.innerHTML = outerHtml;
if (caption.parentNode == bottomParent) {
bottomParent.removeChild(caption);
}
if (caption.parentNode == videoCationsShell) {
videoCationsShell.removeChild(caption);
}
root.appendChild(n);
}
document.addEventListener("fullscreenchange", function () {
if (document.fullscreen) {
toFull();
} else {
changed = false;
checkCaption();
}
});
function addWordMouseListner() {
$(document).on("click", "span.lln-word", function (e) {
console.log('xxxxxxxxxxxxxx');
onWordClick(e);
});
function onWordClick(e) {
let v = GM_getValue("openBing");
if (!v) {
return;
}
if (isFullScreen()) return;
var word = e.target.innerText;
var arr = word.split(/[\s\n]/);
word = arr[arr.length - 1];
var reg = /[^\x00-\x7F]+/
if (reg.test(word)) return;
if (!word) return;
var url = `https://dictionary.cambridge.org/dictionary/english-chinese-simplified/` + word;
//url = `https://www.bing.com/dict/search?q=` + word;
//q=work&FORM=BDVSP6&cc=cn
//url = `https://www.bing.com/dict/search?q=` + word + `&FORM=BDVSP6&cc=cn`;
var win = window.open(url, "Dictionary", "width=750,height=2090 top=0 left=1330");
setTimeout(function () {
var evt = document.createEvent("MouseEvents");
evt.initMouseEvent("mouseup", true, true);
}, 50);
}
/**
$(document).on("click", "span.lln-word.lln-hover-tooltip.top", function(e){
let v = GM_getValue("openBing");
if(!v){
return;
}
if(isFullScreen())return;
var word = e.target.innerText;
var arr = word.split(/[\s\n]/);
word = arr[arr.length-1];
var reg = /[^\x00-\x7F]+/
if(reg.test(word))return;
if(!word)return;
var url = `https://dictionary.cambridge.org/dictionary/english-chinese-simplified/` + word;
url = `https://www.bing.com/dict/search?q=` + word;
//q=work&FORM=BDVSP6&cc=cn
//url = `https://www.bing.com/dict/search?q=` + word + `&FORM=BDVSP6&cc=cn`;
var win = window.open(url, "Dictionary", "width=750,height=2090 top=0 left=1330");
setTimeout(function(){
var evt = document.createEvent("MouseEvents");
evt.initMouseEvent("mouseup", true, true);
},50);
});
*/
}
function isFullScreen() {
return !!(
document.fullscreen ||
document.mozFullScreen ||
document.webkitIsFullScreen ||
document.webkitFullScreen ||
document.msFullScreen
);
}
// 窗口的大小和在屏幕中的位置,具体可以根据自己的显示器调整。
let wo = window.open;
//window.open = open;
let win
function open() {
console.log('win open', arguments);
if (arguments[0].indexOf('youtube.com') > -1) {
return wo.apply(window, arguments);
}
//arguments[2] = `width=750,height=2090 top=0 left=${window.outerWidth + -13}`;
let left = Math.min(screen.width - 750, window.outerWidth + -13);
arguments[1] = "Dictionary";
arguments[2] = `width=750,height=${window.outerHeight} top=0 left=${left}`;
var w;
try {
w = wo.apply(window, arguments);
tryToStopVideo();
if (w) {
win = w;
}
} catch (e) {
}
return w;
}
window.onbeforeunload = function () {
if (!win) return;
win.close();
}
function onFullScreenClick() {
const llnCloseVV = document.querySelector("#llnCloseVV");
if (!llnCloseVV) return;
var evt = document.createEvent("MouseEvents");
evt.initMouseEvent("click", true, true);
var bl = llnCloseVV.dispatchEvent(evt);
}
document.addEventListener("fullscreenchange", function (e) {
if (document.fullscreenElement) {
onFullScreenClick();
} else {
onFullScreenExit();
}
})
function onFullScreenExit() {
const sidebar = document.querySelector(".lln-vv-toggle-in-sidebar");
if (!sidebar) return;
var evt = document.createEvent("MouseEvents");
evt.initMouseEvent("click", true, true);
sidebar.dispatchEvent(evt);
}
function addSubClick() {
$(document).on("click", ".lln-vertical-view-sub .lln-word", function (e) {
let v = GM_getValue("openBing");
if (!v) {
return;
}
var word = e.target.innerText;
var arr = word.split(/[\s\n]/);
word = arr[arr.length - 1];
var reg = /[^\x00-\x7F]+/
if (reg.test(word)) return;
if (!word) return;
var url = `https://www.bing.com/dict/search?q=` + word;
url = `https://dictionary.cambridge.org/dictionary/english-chinese-simplified/` + word
let left = Math.min(screen.width - 750, window.outerWidth + -13);
arguments[2] = `width=750,height=${window.outerHeight} top=0 left=${left}`;
//var win = window.open(url, "Dictionary", "width=750,height=2090 top=0 left=1330");
var win = window.open(url, "Dictionary", `width=750,height=${window.outerHeight} top=0 left=${left}`);
setTimeout(function () {
var evt = document.createEvent("MouseEvents");
evt.initMouseEvent("mouseup", true, true);
}, 50);
});
}
function tryToStopVideo() {
let v = document.querySelector("video");
if (!v) return;
let isPlaying = videoIsPlaying(v);
if (!isPlaying) return;
v.pause();
}
function videoIsPlaying(video) {
return !!(video.currentTime > 0 && !video.paused && !video.ended && video.readyState > 2)
}
window.open = open;
addWordMouseListner();
addSubClick();
})();