您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
mouse-over video thumbnail to preview video by cycling through the storyboard
- // ==UserScript==
- // @name Youtube Storyboard
- // @namespace hbb_works
- // @description mouse-over video thumbnail to preview video by cycling through the storyboard
- // @version 1.3.1r1605101704
- // @include http://*youtube.com*
- // @include https://*youtube.com*
- // @grant none
- // ==/UserScript==
- const INTERVAL = 200; // default interval in milliseconds
- var div;
- var loader;
- var animator;
- var mousePos = { x: 0, y: 0 };
- function dbg() {
- //console.log.apply(console, arguments);
- }
- function getPos(element) {
- var ref = document.body.getBoundingClientRect();
- var rel = element.getBoundingClientRect();
- return {
- left: rel.left - ref.left,
- top: rel.top - ref.top
- };
- }
- function setMousePos(event) {
- mousePos = { x: event.pageX, y: event.pageY };
- }
- (function init() {
- dbg("init youtube storyboard");
- document.addEventListener('mouseover', onMouseOver, false);
- loader = document.createElement('img');
- div = document.createElement('div');
- div.id = 'storyboardPlayer';
- div.style.display = 'none';
- div.style.position = 'absolute';
- div.style.zIndex = 99;
- var a = document.createElement('a');
- var innerDiv = document.createElement('div');
- innerDiv.style.height = '100%';
- innerDiv.style.width = '100%';
- a.appendChild(innerDiv);
- div.appendChild(a);
- document.getElementById('page') .appendChild(div);
- div.addEventListener('mouseout', function () {
- animator.stop();
- });
- document.addEventListener('mousemove', function (event) {
- setMousePos(event);
- });
- })();
- function onMouseOver(event) {
- setMousePos(event);
- var target = event.target;
- var src = target.src;
- if (target.nodeName == 'IMG' && src && (/default\.jpg/.exec(src) || /0\.jpg/.exec(src)))
- {
- if (animator) {
- animator.stop();
- }
- animator = new Animator(target);
- animator.getStoryboardSpecs(function () {
- animator.start();
- });
- }
- }
- var Animator = function (target) {
- this.id = Math.floor(Math.random() * 10000);
- dbg("new " + this.id);
- this.target = target;
- this.frame = 0;
- this.sheet = 0;
- this.interval = INTERVAL;
- this.src = target.src
- // get video id
- this.v = (/vi\/(.*?)\//.exec(this.src) || []) [1];
- if (this.v) {
- this.width = target.width;
- this.height = target.height;
- }
- var p = getPos(target);
- div.style.top = p.top + 'px';
- div.style.left = p.left + 'px';
- div.style.width = this.width + 'px';
- div.style.height = this.height + 'px';
- this.x = p.left;
- this.y = p.top;
- // adjustments for small thumbnails in list of related videos
- if (target.parentElement.classList.contains('yt-uix-simple-thumb-wrap')) {
- var spanHeight = target.parentElement.getBoundingClientRect().height;
- div.style.top = p.top + ((target.height - spanHeight) / 2) + 'px';
- div.style.height = spanHeight + 'px';
- this.height = spanHeight;
- }
- var parent = target.parentElement;
- while (parent.tagName != 'A') {
- parent = parent.parentElement;
- }
- div.children[0].href = parent.href;
- div.style.backgroundRepeat = 'no-repeat';
- div.style.backgroundPosition = 'center';
- div.style.backgroundSize = null;
- // loading indicator
- div.style.backgroundImage ='url(\'\')';
- div.style.display = null;
- };
- Animator.prototype.parseSpec = function (storyboard_spec) {
- dbg("parseSpec " + this.id);
- var lines = storyboard_spec.split('|');
- var q;
- if (!this.quality) {
- this.quality = - 1;
- do {
- this.quality++;
- q = lines[this.quality + 1].split('#');
- } while (parseInt(q[0], 10) < this.width && this.quality + 2 < lines.length);
- }
- var q = lines[this.quality + 1].split('#');
- var s = {
- url: lines[0].replace('$L', this.quality) .replace('$N', q[6]),
- width: parseInt(q[0], 10),
- height: parseInt(q[1], 10),
- count: parseInt(q[2], 10),
- cols: parseInt(q[3], 10),
- rows: parseInt(q[4], 10),
- sigh: q[7]
- };
- s.sheetSize = s.cols * s.rows;
- s.sheetCount = ((s.count / s.sheetSize) | 0) + 1; // bitwise OR to loose decimals
- s.countOnLastSheet = ((s.count - 1) % s.sheetSize) + 1;
- return this.spec = s;
- };
- Animator.prototype.loadImage = function (callback) {
- dbg("loadImage " + this.id);
- var onLoad = (function () {
- div.style.backgroundImage = 'url(' + loader.src + ')';
- loader.removeEventListener('load', onLoad);
- callback.apply(this);
- }).bind(this);
- loader.addEventListener('load', onLoad);
- loader.src = this.spec.url.replace('$M', this.sheet) + '?sigh=' + this.spec.sigh;
- };
- Animator.prototype.getStoryboardSpecs = function (callback, fromWatch) {
- dbg("getStoryboardSpecs (fromWatch: " + fromWatch + ") " + this.id);
- this.xhr = new XMLHttpRequest();
- this.xhr.onload = (function () {
- if (this.isMouseOver()) {
- if (fromWatch) {
- var spec = (/"storyboard_spec":\s*(".*?")/.exec(this.xhr.responseText) || []) [1];
- }
- else {
- var spec = (/&storyboard_spec=(.*?)&/.exec(this.xhr.responseText) || []) [1];
- }
- if (spec) {
- if (fromWatch) {
- spec = eval(spec); // remove backslashes
- }
- this.parseSpec(decodeURIComponent(spec));
- callback.apply(this);
- }
- else if (!fromWatch) {
- this.getStoryboardSpecs(callback, true);
- }
- else {
- div.style.background = null;
- }
- }
- else {
- this.stop();
- }
- }).bind(this);
- if (fromWatch) {
- this.xhr.open('GET', '/watch?v=' + this.v, true);
- }
- else {
- this.xhr.open('GET', '/get_video_info?video_id=' + this.v, true);
- }
- this.xhr.send();
- };
- Animator.prototype.start = function () {
- dbg("start " + this.id);
- this.frame = 0;
- this.sheet = 0;
- var lastTick;
- var next = function () {
- dbg("next " + this.id + " token " + this.token);
- if (this.isMouseOver()) {
- if (this.spec.fit == 'height') {
- var w = this.spec.width * this.height / this.spec.height;
- var offset = (this.width - w) / 2;
- div.style.backgroundPosition = -((this.frame % this.spec.cols) * w - offset) + 'px ' +
- -(((this.frame / this.spec.cols) | 0) * this.height) + 'px'; // bitwise OR with 0 to loose decimals
- }
- else {
- var h = this.spec.height * this.width / this.spec.width;
- var offset = (this.height - h) / 2;
- div.style.backgroundPosition = -((this.frame % this.spec.cols) * this.width) + 'px ' +
- -(((this.frame / this.spec.cols) | 0) * h - offset) + 'px';
- }
- this.frame++;
- if ((this.frame == this.spec.sheetSize) || (this.sheet == this.spec.sheetCount - 1 && this.frame == this.spec.countOnLastSheet))
- {
- clearInterval(this.token);
- dbg("clear " + this.id + " token " + this.token);
- this.loadImage(function () {
- this.frame = 0;
- this.sheet = (this.sheet + 1) % this.spec.sheetCount;
- this.token = setInterval(next, this.interval);
- dbg("set " + this.id + " token " + this.token);
- });
- }
- }
- else {
- this.stop();
- }
- }.bind(this);
- this.loadImage(function () {
- if (this.isMouseOver()) {
- if ((this.width / this.height) <= (this.spec.width / this.spec.height)) {
- this.spec.fit = 'height';
- div.style.backgroundSize = ((this.spec.width * this.height / this.spec.height) * this.spec.cols) + 'px ' +
- (this.height * this.spec.rows) + 'px';
- }
- else {
- this.spec.fit = 'width';
- div.style.backgroundSize = (this.width * this.spec.cols) + 'px ' +
- ((this.spec.height * this.width / this.spec.width) * this.spec.rows) + 'px';
- }
- this.token = setInterval(next, this.interval);
- dbg("set " + id + " token " + this.token);
- }
- else {
- this.stop();
- }
- });
- };
- Animator.prototype.stop = function () {
- dbg("stop " + this.id);
- if (!this.stopped) {
- clearInterval(this.token);
- dbg("clear " + this.id + " token " + this.token);
- div.style.display = 'none';
- }
- this.stopped = true;
- };
- Animator.prototype.isMouseOver = function () {
- return mousePos.x >= this.x && mousePos.x <= this.x + this.width &&
- mousePos.y >= this.y && mousePos.y <= this.y + this.height;
- };