- // ==UserScript==
- // @name * Lyric FullScreen Columnizer
- // @name:ja * Lyric FullScreen Columnizer
- // @name:zh-CN * Lyric FullScreen Columnizer
- // @namespace knoa.jp
- // @description It offers a full-width and columnized lyric view on major lyric services. No more scrolling while singing, playing the piano, or guitar.
- // @description:ja 大手歌詞サイトの歌詞を、横幅いっぱいの複数カラム表示に。歌いながら、ピアノやギターを弾きながら、スクロールしなくてもいいんです。
- // @description:zh-CN 将大型歌词网站的歌词显示为宽度最大的多列。一边唱歌,一边弹钢琴和吉他,不用滚动。
- // @include https://www.google.*/*Lyric*
- // @include https://www.google.*/*%E6%AD%8C%E8%A9%9E*
- // @include https://www.google.*/*%E6%AD%8C%E8%AF%8D*
- // @include https://www.azlyrics.com/lyrics/*
- // @include https://genius.com/*
- // @include https://www.lyrics.com/lyric/*
- // @include https://j-lyric.net/artist/*
- // @include http*://www.kget.jp/lyric/*
- // @include https://www.uta-net.com/song/*
- // @include https://utaten.com/lyric/*
- // @noframes
- // @version 2.2.0
- // @grant none
- // ==/UserScript==
-
- (function(){
- const SCRIPTID = 'LyricFullScreenColumnizer';
- const SCRIPTNAME = '* Lyric FullScreen Columnizer';
- const DEBUG = false;/*
- [update]
- Now available on Genius Lyrics.
-
- [possible]
- lyrics.com はpreなので単語が切れる。br挿入してnormalテキストにすれば解決するが。
- うたまっぷ は大手だがHTMLが古いのでいまのところ対応しない。
-
- [acknowledgement]
- This script is originally dedicated to Milky Queen, for singing freely with her guitar playing.
- 🌾👑 https://twitter.com/milkyqueen_idol
- */
- if(window === top) console.time(SCRIPTID);
- const MS = 1, SECOND = 1000*MS, MINUTE = 60*SECOND, HOUR = 60*MINUTE, DAY = 24*HOUR, WEEK = 7*DAY, MONTH = 30*DAY, YEAR = 365*DAY;
- const sites = {
- google: {
- /* it doesn't detect url with "lyric" or something here, but @include meta tag does */
- url: /^https:\/\/www\.google\.[^/]+\//,
- targets: {
- header: () => $('#sfcnt'),
- lyricBody: () => $('[data-lyricid]'),
- },
- actions: {
- beforeColumnize: () => $('g-more-link [aria-expanded="false"]', e => e.click()),
- }
- },
- azlyrics: {
- url: /^https:\/\/www\.azlyrics\.com\/lyrics\//,
- targets: {
- header: () => $('.lboard-wrap'),
- lyricBody: () => $('.main-page br + br + div'),
- },
- },
- genius: {
- url: /^https:\/\/genius\.com\//,
- targets: {
- header: () => $('.header'),
- lyricBody: () => $('.lyrics'),
- },
- },
- lyrics: {
- url: /^https:\/\/www\.lyrics\.com\/lyric\//,
- targets: {
- header: () => $('#content-top'),
- lyricBody: () => $('#lyric-body-text'),
- },
- },
- jlyric: {
- url: /^https:\/\/j-lyric\.net\/artist\//,
- targets: {
- header: () => $('#ttb'),
- lyricBody: () => $('#Lyric'),
- },
- },
- kget: {
- url: /^https?:\/\/www\.kget\.jp\/lyric\//,
- targets: {
- header: () => $('#searchbar-wrap'),
- lyricBody: () => $('#lyric-trunk'),
- },
- },
- utanet: {
- url: /^https:\/\/www\.uta-net\.com\/song\//,
- targets: {
- header: () => $('#global_header'),
- lyricBody: () => $('#kashi_area'),
- },
- },
- utaten: {
- url: /^https:\/\/utaten\.com\/lyric\//,
- targets: {
- header: () => $('body > header'),
- lyricBody: () => $('.lyricBody'),
- },
- },
- };
- let site;
- let elements = {};
- const core = {
- initialize: function(){
- elements.html = document.documentElement;
- elements.html.classList.add(SCRIPTID);
- site = core.getSite(sites);
- if(site){
- core.ready();
- core.addStyle('style');
- core.addStyle('style-' + site.key);
- }
- },
- ready: function(){
- core.getTargets(site.targets).then(() => {
- log("I'm ready.");
- core.bindKeys();
- }).catch(e => {
- console.error(`${SCRIPTID}:`, e);
- });
- },
- bindKeys: function(){
- const {header, lyricBody} = elements;
- const beforeLyricBody = elements.lyricBody?.previousElementSibling;
- const parentOfLyricBody = elements.lyricBody?.parentNode;
- window.addEventListener('keydown', e => {
- if(['input', 'textarea'].includes(e.target.localName) || e.target.isContentEditable) return;
- if(e.ctrlKey || e.altKey || e.shiftKey || e.metaKey) return;
- console.log(SCRIPTID, e.key);
- switch(e.key){
- /* columnize */
- case('1'):
- case('2'):
- case('3'):
- case('4'):
- case('5'):
- case('6'):
- case('7'):
- case('8'):
- case('9'):
- document.body.classList.add(SCRIPTID);
- if(site.actions?.beforeColumnize) site.actions.beforeColumnize();
- if(document.fullscreenElement === null) header.after(lyricBody);
- lyricBody.dataset.columns = e.key;
- e.preventDefault();
- break;
- /* reset to default */
- case('0'):
- case('Escape'):
- document.body.classList.remove(SCRIPTID);
- if(beforeLyricBody) beforeLyricBody.after(lyricBody);
- else parentOfLyricBody.prepend(lyricBody);
- delete lyricBody.dataset.columns;
- e.preventDefault();
- break;
- /* browser's fullscreen */
- case('f'):
- if(document.fullscreenElement === null){
- document.body.classList.add(SCRIPTID);
- if(site.actions?.beforeColumnize) site.actions.beforeColumnize();
- if(lyricBody.dataset.columns === undefined) lyricBody.dataset.columns = '1';
- lyricBody.requestFullscreen();
- }
- else document.exitFullscreen();
- e.preventDefault();
- break;
- }
- }, true);
- /* fire the reset event on fullscreen exit */
- window.addEventListener('fullscreenchange', e => {
- if(document.fullscreenElement) return;
- else window.dispatchEvent(new KeyboardEvent('keydown', {key: 'Escape'}));
- });
- },
- getSite: function(sites){
- Object.keys(sites).forEach(key => sites[key].key = key);
- let key = Object.keys(sites).find(key => sites[key].url.test(location.href));
- if(key === undefined) return log('Doesn\'t match any sites:', location.href);
- else return sites[key];
- },
- getTarget: function(selector, retry = 10, interval = 1*SECOND){
- const key = selector.name;
- const get = function(resolve, reject){
- let selected = selector();
- if(selected === null || selected.length === 0){
- if(--retry) return log(`Not found: ${key}, retrying... (${retry})`), setTimeout(get, interval, resolve, reject);
- else return reject(new Error(`Not found: ${selector.name}, I give up.`));
- }else{
- if(selected.nodeType === Node.ELEMENT_NODE) selected.dataset.selector = key;/* element */
- else selected.forEach((s) => s.dataset.selector = key);/* elements */
- elements[key] = selected;
- resolve(selected);
- }
- };
- return new Promise(function(resolve, reject){
- get(resolve, reject);
- });
- },
- getTargets: function(selectors, retry = 10, interval = 1*SECOND){
- return Promise.all(Object.values(selectors).map(selector => core.getTarget(selector, retry, interval)));
- },
- addStyle: function(name = 'style', d = document){
- if(html[name] === undefined) return;
- if(d.head){
- let style = createElement(html[name]()), id = SCRIPTID + '-' + name, old = d.getElementById(id);
- style.id = id;
- d.head.appendChild(style);
- if(old) old.remove();
- }
- else{
- let observer = observe(d.documentElement, function(){
- if(!d.head) return;
- observer.disconnect();
- core.addStyle(name);
- });
- }
- },
- };
- const html = {
- style: () => `
- <style type="text/css">
- /* maximize lyricBody */
- [data-selector="lyricBody"][data-columns]{
- width: 100vw;
- padding: 2em 1em 2em 2em;
- margin: 0;
- box-sizing: border-box;
- }
- /* columnize */
- [data-selector="lyricBody"][data-columns="1"]{columns: 1}
- [data-selector="lyricBody"][data-columns="2"]{columns: 2}
- [data-selector="lyricBody"][data-columns="3"]{columns: 3}
- [data-selector="lyricBody"][data-columns="4"]{columns: 4}
- [data-selector="lyricBody"][data-columns="5"]{columns: 5}
- [data-selector="lyricBody"][data-columns="6"]{columns: 6}
- [data-selector="lyricBody"][data-columns="7"]{columns: 7}
- [data-selector="lyricBody"][data-columns="8"]{columns: 8}
- [data-selector="lyricBody"][data-columns="9"]{columns: 9}
- /* no distracting elements */
- [data-selector="lyricBody"][data-columns] ~ *{
- display: none;
- }
- </style>
- `,
- 'style-google': () => `
- <style type="text/css">
- body.${SCRIPTID} [data-selector="lyricBody"]{
- background: white;/* i don't know why it is required on fullscreen */
- }
- body.${SCRIPTID} .OULBYb/* SO FRAGILE!! */{
- display: none;
- }
- body.${SCRIPTID} [role="contentinfo"]{
- display: block;
- }
- </style>
- `,
- 'style-azlyrics': () => `
- <style type="text/css">
- body.${SCRIPTID} [data-selector="lyricBody"]{
- background: rgb(221, 221, 238);/* i don't know why it is required on fullscreen */
- }
- body.${SCRIPTID} .navbar-bottom,
- body.${SCRIPTID} .navbar-bottom ~ div{
- display: block;
- }
- </style>
- `,
- 'style-genius': () => `
- <style type="text/css">
- body.${SCRIPTID} [data-selector="lyricBody"]{
- background: white;/* i don't know why it is required on fullscreen */
- }
- body.${SCRIPTID} .page_footer{
- display: block;
- }
- </style>
- `,
- 'style-lyrics': () => `
- <style type="text/css">
- body.${SCRIPTID} #main{
- width: 100vw;
- margin: 20px 0 0;
- max-width: 100vw;
- padding: 0;
- }
- body.${SCRIPTID} [data-selector="lyricBody"]{
- white-space: pre-wrap;
- font-family: 'Droid Sans',sans-serif;
- font-weight: 400;
- font-size: 18px;
- line-height: 26px;
- background: white;/* i don't know why it is required on fullscreen */
- }
- body.${SCRIPTID} footer{
- display: block;
- }
- </style>
- `,
- 'style-jlyric': () => `
- <style type="text/css">
- body.${SCRIPTID} [data-selector="lyricBody"]{
- margin: 0 !important;
- background: white;/* i don't know why it is required on fullscreen */
- }
- body.${SCRIPTID} #ftb{
- display: block;
- }
- </style>
- `,
- 'style-kget': () => `
- <style type="text/css">
- body.${SCRIPTID} [data-selector="lyricBody"]{
- font-size: 123.1%;
- font-family: "Hiragino Mincho ProN", Meiryo, "MS PMincho", serif;
- background: white;/* i don't know why it is required on fullscreen */
- }
- body.${SCRIPTID} [data-selector="lyricBody"] > a{
- display: none;
- }
- body.${SCRIPTID} #footer-wrap{
- display: block;
- }
- </style>
- `,
- 'style-utanet': () => `
- <style type="text/css">
- body.${SCRIPTID} [data-selector="lyricBody"]{
- font-size: 15px;
- background: white;/* i don't know why it is required on fullscreen */
- }
- body.${SCRIPTID} #footer_map,
- body.${SCRIPTID} #footer_bottom{
- display: block;
- }
- </style>
- `,
- 'style-utaten': () => `
- <style type="text/css">
- body.${SCRIPTID} footer{
- display: block !important;
- }
- body.${SCRIPTID} footer > aside{
- display: none;
- }
- body.${SCRIPTID}{
- background: #343330;
- }
- </style>
- `,
- };
- const $ = function(s, f = undefined){
- let target = document.querySelector(s);
- if(target === null) return null;
- return f ? f(target) : target;
- };
- const $$ = function(s, f = undefined){
- let targets = document.querySelectorAll(s);
- return f ? f(targets) : targets;
- };
- const createElement = function(html = '<div></div>'){
- let outer = document.createElement('div');
- outer.insertAdjacentHTML('afterbegin', html);
- return outer.firstElementChild;
- };
- const log = function(){
- if(typeof DEBUG === 'undefined') return;
- console.log(...log.build(new Error(), ...arguments));
- };
- log.build = function(error, ...args){
- let l = log.last = log.now || new Date(), n = log.now = new Date();
- let line = log.format.getLine(error), callers = log.format.getCallers(error);
- //console.log(error.stack);
- return [SCRIPTID + ':',
- /* 00:00:00.000 */ n.toLocaleTimeString() + '.' + n.getTime().toString().slice(-3),
- /* +0.000s */ '+' + ((n-l)/1000).toFixed(3) + 's',
- /* :00 */ ':' + line,
- /* caller.caller */ (callers[2] ? callers[2] + '() => ' : '') +
- /* caller */ (callers[1] || '') + '()',
- ...args
- ];
- };
- log.formats = [{
- name: 'Firefox Scratchpad',
- detector: /MARKER@Scratchpad/,
- getLine: (e) => e.stack.split('\n')[1].match(/([0-9]+):[0-9]+$/)[1],
- getCallers: (e) => e.stack.match(/^[^@]*(?=@)/gm),
- }, {
- name: 'Firefox Console',
- detector: /MARKER@debugger/,
- getLine: (e) => e.stack.split('\n')[1].match(/([0-9]+):[0-9]+$/)[1],
- getCallers: (e) => e.stack.match(/^[^@]*(?=@)/gm),
- }, {
- name: 'Firefox Greasemonkey 3',
- detector: /\/gm_scripts\//,
- getLine: (e) => e.stack.split('\n')[1].match(/([0-9]+):[0-9]+$/)[1],
- getCallers: (e) => e.stack.match(/^[^@]*(?=@)/gm),
- }, {
- name: 'Firefox Greasemonkey 4+',
- detector: /MARKER@user-script:/,
- getLine: (e) => e.stack.split('\n')[1].match(/([0-9]+):[0-9]+$/)[1] - 500,
- getCallers: (e) => e.stack.match(/^[^@]*(?=@)/gm),
- }, {
- name: 'Firefox Tampermonkey',
- detector: /MARKER@moz-extension:/,
- getLine: (e) => e.stack.split('\n')[1].match(/([0-9]+):[0-9]+$/)[1] - 2,
- getCallers: (e) => e.stack.match(/^[^@]*(?=@)/gm),
- }, {
- name: 'Chrome Console',
- detector: /at MARKER \(<anonymous>/,
- getLine: (e) => e.stack.split('\n')[2].match(/([0-9]+):[0-9]+\)?$/)[1],
- getCallers: (e) => e.stack.match(/[^ ]+(?= \(<anonymous>)/gm),
- }, {
- name: 'Chrome Tampermonkey',
- detector: /at MARKER \(chrome-extension:.*?\/userscript.html\?name=/,
- getLine: (e) => e.stack.split('\n')[2].match(/([0-9]+):[0-9]+\)?$/)[1] - 1,
- getCallers: (e) => e.stack.match(/[^ ]+(?= \(chrome-extension:)/gm),
- }, {
- name: 'Chrome Extension',
- detector: /at MARKER \(chrome-extension:/,
- getLine: (e) => e.stack.split('\n')[2].match(/([0-9]+):[0-9]+\)?$/)[1],
- getCallers: (e) => e.stack.match(/[^ ]+(?= \(chrome-extension:)/gm),
- }, {
- name: 'Edge Console',
- detector: /at MARKER \(eval/,
- getLine: (e) => e.stack.split('\n')[2].match(/([0-9]+):[0-9]+\)$/)[1],
- getCallers: (e) => e.stack.match(/[^ ]+(?= \(eval)/gm),
- }, {
- name: 'Edge Tampermonkey',
- detector: /at MARKER \(Function/,
- getLine: (e) => e.stack.split('\n')[2].match(/([0-9]+):[0-9]+\)$/)[1] - 4,
- getCallers: (e) => e.stack.match(/[^ ]+(?= \(Function)/gm),
- }, {
- name: 'Safari',
- detector: /^MARKER$/m,
- getLine: (e) => 0,/*e.lineが用意されているが最終呼び出し位置のみ*/
- getCallers: (e) => e.stack.split('\n'),
- }, {
- name: 'Default',
- detector: /./,
- getLine: (e) => 0,
- getCallers: (e) => [],
- }];
- log.format = log.formats.find(function MARKER(f){
- if(!f.detector.test(new Error().stack)) return false;
- //console.log('////', f.name, 'wants', 0/*the exact line number here*/, '\n' + new Error().stack);
- return true;
- });
- core.initialize();
- if(window === top) console.timeEnd(SCRIPTID);
- })();