AbemaTV Screen Comment Scroller

AbemaTV のコメントをニコニコ風にスクロールさせます。

ของเมื่อวันที่ 05-10-2017 ดู เวอร์ชันล่าสุด

// ==UserScript==
// @name        AbemaTV Screen Comment Scroller
// @namespace   knoa.jp
// @description AbemaTV のコメントをニコニコ風にスクロールさせます。
// @include     https://abema.tv/*
// @version     1.3.7
// @grant       none
// ==/UserScript==

// console.log('AbemaTV? => hireMe()');
(function(){
  const SCRIPTNAME = 'ScreenCommentScroller';
  const DEBUG = false;//アベマの仕様変更に対応しました。
  // delete localStorage['ScreenCommentScroller-configs'];
  if(window === top) console.time(SCRIPTNAME);
  const CONFIGS = [
    /*スクロールコメント*/
    {KEY: 'color',       DEFAULT: '#ffffff', TYPE: 'string'},/*色*/
    {KEY: 'ocolor',      DEFAULT: '#000000', TYPE: 'string'},/*縁取り色*/
    {KEY: 'owidth',      DEFAULT: 0.05,      TYPE: 'float' },/*縁取りの太さ(比率)*/
    {KEY: 'maxlines',    DEFAULT: 10,        TYPE: 'int'   },/*最大行数*/
    {KEY: 'linemargin',  DEFAULT: 0.2,       TYPE: 'float' },/*行間(比率)*/
    {KEY: 'opacity',     DEFAULT: 0.50,      TYPE: 'float' },/*不透明度*/
    {KEY: 'hopacity',    DEFAULT: 0.50,      TYPE: 'float' },/*不透明度(マウスオーバー時)*/
    /*一覧コメント*/
    {KEY: 'lt_opacity',  DEFAULT: 0.75,      TYPE: 'float' },/*文字の不透明度*/
    {KEY: 'lt_hopacity', DEFAULT: 1.00,      TYPE: 'float' },/*文字の不透明度(マウスオーバー時)*/
    {KEY: 'lb_opacity',  DEFAULT: 0.25,      TYPE: 'float' },/*背景の不透明度*/
    {KEY: 'lb_hopacity', DEFAULT: 0.50,      TYPE: 'float' },/*背景の不透明度(マウスオーバー時)*/
    /*アニメーション*/
    {KEY: 'duration',    DEFAULT: 5,         TYPE: 'float' },/*横断にかける秒数*/
    {KEY: 'fps',         DEFAULT: 60,        TYPE: 'int'   },/*秒間コマ数*/
  ];
  const AINTERVAL = 5;/*AbemaTVのコメント取得間隔の仕様値*/
  const ADELAYS = {/*AbemaTVのコメント取得時の投稿時刻を(AINTERVAL)まで用意しておく*/
    '今': 0,
    '1秒前': 1,
    '2秒前': 2,
    '3秒前': 3,
    '4秒前': 4,
    '5秒前': 5,
  };
  /* サイト定義 */
  let site = {
    targets: [
      /* 構造 */
      function header(){let header = document.querySelector('body > div > div > header'); return (header) ? site.use(header) : null;},
      function footer(){let fullscreen = document.querySelector('button[aria-label="フルスクリーン表示"]'); return (fullscreen) ? site.use(fullscreen.parentNode.parentNode) : null;},
      function board(){let board = document.querySelector('div[aria-hidden] form + div > div'); return (board) ? site.use(board) : null;},
      function screen(){let loading = document.querySelector('img[src="/images/misc/feed_loading.gif"]'); return (loading) ? site.use(loading.parentNode.parentNode) : null;},
      /* ペイン */
      function commentPane(){let form = document.querySelector('form:not([role="search"])'); return (form) ? site.use(form.parentNode.parentNode) : null;},
      function channelPane(){let list = document.querySelector('use[*|href^="/images/icons/list.svg"]'); return (list) ? site.use(list.parentNode.parentNode.parentNode.parentNode.nextElementSibling) : null;},
      function programPane(){let list = document.querySelector('use[*|href^="/images/icons/list.svg"]'); return (list) ? site.use(list.parentNode.parentNode.parentNode.parentNode.nextElementSibling.nextElementSibling) : null;},
      /* ボタン */
      function channelButtons(){let list = document.querySelector('use[*|href^="/images/icons/list.svg"]'); return (list) ? site.use(list.parentNode.parentNode.parentNode.parentNode) : null;},
      function channelButton(){let list = document.querySelector('use[*|href^="/images/icons/list.svg"]'); return (list) ? site.use(list.parentNode.parentNode) : null;},
      function commentButton(){let svg = document.querySelector('use[*|href^="/images/icons/comment.svg"]'); return (svg) ? site.use(svg.parentNode.parentNode) : null;},
      function programButton(){let button = document.querySelector('button[aria-label="フルスクリーン表示"] + div + div > div > div'); return (button) ? site.use(button) : null;},
      function fullscreenButton(){let fullscreen = document.querySelector('button[aria-label="フルスクリーン表示"]'); return (fullscreen) ? site.use(fullscreen) : null;},
      function closer(){let commentForm = document.querySelector('form:not([role="search"])'); return (commentForm) ? site.use(commentForm.parentNode.parentNode.nextElementSibling) : null;},
      /* 表示要素 */
      function audience(){let loading = document.querySelector('img[src="/images/misc/feed_loading.gif"]'); return (loading) ? site.use(loading.parentNode.parentNode.firstChild.firstChild) : null;},
      function programName(){let name = document.querySelector('button[aria-label="フルスクリーン表示"] + div + div div > p > span > span:last-child'); return (name) ? site.use(name) : null;},
      function notice(){let header = document.querySelector('header'); return (header) ? site.use(header.nextElementSibling) : null;},
    ],
    getComments: function(target){return (target.querySelectorAll) ? target.querySelectorAll('div[aria-hidden] form + div > div > div > div > div > p:first-child') : null},
    use: function use(target){
      const cid = 'selectorId'/*camelCase*/, sid = 'selector-id'/*snake-case*/;
      target.dataset[cid] = use.caller.name;
      selectors[use.caller.name] = `${target.localName}[data-${sid}="${use.caller.name}"]`;
      elements[use.caller.name] = target;
      return true;
    },
  };
  /* 処理本体 */
  let elements = {}, selectors = {}, canvas, context, lines = [], fontsize, interval, configButton, configPanel, configs = {}, style;
  let core = {
    /* 初期化 */
    initialize: function(){
      let previousUrl = '';
      /* 一度だけ */
      html = document.documentElement;
      core.config.read();
      window.addEventListener('resize', setTimeout.bind(null, core.modify, 1000));
      /* URLの変化を見守る */
      setInterval(function(){
        if(location.href === previousUrl) return;/*URLが変わってない*/
        /* テレビ視聴ページ */
        if(location.href.startsWith('https://abema.tv/now-on-air/')){
          if(previousUrl.startsWith('https://abema.tv/now-on-air/')) return;/*チャンネルを変えただけ*/
          html.classList.add(SCRIPTNAME);
          core.ready();
        /* テレビ視聴ページではない */
        }else{
          html.classList.remove(SCRIPTNAME);
        }
        previousUrl = location.href;
      }, 1000);
    },
    /* URLが変わるたびに呼ぶ */
    ready: function(e){
      /* 必要な要素が出揃うまで粘る */
      for(let i = 0; site.targets[i]; i++) if(site.targets[i]() === null) return setTimeout(core.ready, 1000);
      core.addStyle();
      core.config.createButton();
      /* 開けるようになったら自動で開く */
      observe(elements.commentButton, function(records){
        if(getComputedStyle(this).cursor === 'pointer'){
          this.click();
        }
      }, {attributes: true});
      /* 設定画面を用意する */
      core.config.createButton();
      /* コメントをスクロールさせるCanvasの設置 */
      /* (描画処理の軽さは HTML5 Canvas, CSS Position Left, CSS Transition の順) */
      core.createCanvas();
      /* メイン処理 */
      core.listenComments();
      core.scrollComments();
    },
    /* canvas作成 */
    createCanvas: function(){
      if(canvas) return;
      canvas = document.createElement('canvas');
      canvas.id = SCRIPTNAME;
      elements.screen.appendChild(canvas);
      context = canvas.getContext('2d');
      core.modify();
    },
    /* スクリーンサイズに変化があればcanvasも変化させる */
    modify: function(){
      canvas.width = elements.screen.offsetWidth;
      canvas.height = elements.screen.offsetHeight;
      fontsize = (canvas.height / configs.maxlines) / (1 + configs.linemargin);
      context.font = 'bold ' + (fontsize) + 'px sans-serif';
      context.fillStyle = configs.color;
      context.strokeStyle = configs.ocolor;
      context.lineWidth = fontsize * configs.owidth;
    },
    /* コメントの新規追加を見守る */
    listenComments: function(){
      if(elements.board.isListening) return;
      elements.board.isListening = true;
      elements.board.addEventListener('DOMNodeInserted', function(e){
        let comments = site.getComments(e.target);
        if(!comments || !comments.length) return;/*新着コメントの追加でなければ終了*/
        /*投稿経過時間に合わせた時間差を付けることで自然に流す*/
        let earliest = ADELAYS[comments[comments.length - 1].nextElementSibling.textContent];/*同時取得の中で最初に投稿されたコメントの経過時間*/
        if(earliest === undefined) earliest = AINTERVAL;
        for(let i = 0; comments[i]; i++){
          let current = ADELAYS[comments[i].nextElementSibling.textContent];
          if(current === undefined) current = AINTERVAL;
          window.setTimeout(function(){
            core.attachComment(comments[i]);
          }, 1000 * (earliest  - current));
        }
      });
    },
    /* コメントが追加されるたびにスクロールキューに追加 */
    attachComment: function(comment){
      let record = {};
      record.text = comment.textContent;/*流れる文字列*/
      record.width = context.measureText(record.text).width;/*文字列の幅*/
      record.ppms = (canvas.width + record.width) / (configs.duration * 1000);/*ミリ秒あたり移動距離*/
      record.start = Date.now();/*開始時刻*/
      record.reveal = record.start + (record.width / record.ppms);/*文字列が右端から抜ける時刻*/
      record.touch = record.start + (canvas.width / record.ppms);/*文字列が左端に触れる時刻*/
      record.end = record.start + (configs.duration * 1000);/*終了時刻*/
      record.left = canvas.width;/*左端からの距離(描画位置)*/
      /* 追加されたコメントをどの行に流すかを決定する */
      for(let i=0; i < configs.maxlines; i++){
        let length = lines[i] ? lines[i].length : 0;/*同じ行に詰め込まれているコメント数*/
        switch(true){
          /* 行がなければ行を追加して流す */
          case(length === 0):
            lines[i] = [];
          /* ひとつ先行するコメントより遅い(短い)文字列なら、現時点で先行コメントがすでに右端から抜けていれば流す */
          case(record.ppms < lines[i][length - 1].ppms && lines[i][length - 1].reveal < record.start):
          /* ひとつ先行するコメントより速い(長い)文字列なら、左端に触れる瞬間までに先行コメントが終了するなら流す */
          case(lines[i][length - 1].ppms < record.ppms && lines[i][length - 1].end < record.touch):
            break;/*条件に当てはまればswitch文を抜けて行に追加*/
          default:
            continue;/*条件に当てはまらなければforループを回して次の行に入れられるかの判定へ*/
        }
        record.top = ((canvas.height / configs.maxlines) * i) + fontsize;
        lines[i].push(record);
        break;
      }
    },
    /* FPSタイマー駆動 */
    scrollComments: function(){
      if(interval) clearInterval(interval);
      interval = window.setInterval(function(){
        context.clearRect(0, 0, canvas.width, canvas.height);
        /* Canvas描画 */
        let now = Date.now();
        for(let i=0; lines[i]; i++){
          for(let j=0; lines[i][j]; j++){
            /* 視認性を向上させるスクロール文字の縁取りは、幸いにもパフォーマンスにほぼ影響しない */
            context.strokeText(lines[i][j].text, lines[i][j].left, lines[i][j].top);
            context.fillText(lines[i][j].text, lines[i][j].left, lines[i][j].top);
            /* 次の描画位置を計算 */
            lines[i][j].left = canvas.width - ((now - lines[i][j].start) * lines[i][j].ppms);
          }
          if(lines[i][0] && lines[i][0].end < now) lines[i].shift();
        }
      }, 1000 / configs.fps);
    },
    /* 設定 */
    config: {
      read: function(){
        /* 保存済みの設定を読む */
        let ls = localStorage[SCRIPTNAME + '-configs'];
        if(ls) configs = JSON.parse(ls);
        /* 未定義項目をデフォルト値で上書きしていく */
        for(let i = 0; CONFIGS[i]; i++) if(configs[CONFIGS[i].KEY] === undefined) configs[CONFIGS[i].KEY] = CONFIGS[i].DEFAULT;
      },
      save: function(new_config){
        /* CONFIGSを元に文字列を型評価して値を格納していく */
        for(let i = 0; CONFIGS[i]; i++){
          /* 値がなければデフォルト値 */
          if(new_config[CONFIGS[i].KEY] === ""){
            configs[CONFIGS[i].KEY] = CONFIGS[i].DEFAULT;
            continue;
          }
          switch(CONFIGS[i].TYPE){
            case 'int':
              configs[CONFIGS[i].KEY] = parseInt(new_config[CONFIGS[i].KEY]);
              break;
            case 'float':
              configs[CONFIGS[i].KEY] = parseFloat(new_config[CONFIGS[i].KEY]);
              break;
            case 'string':
            default:
              configs[CONFIGS[i].KEY] = new_config[CONFIGS[i].KEY];
              break;
          }
        }
        localStorage[SCRIPTNAME + '-configs'] = JSON.stringify(configs);
      },
      createButton: function(){
        if(configButton) return;
        /* フルスクリーンボタンを元に設定ボタンを追加する */
        configButton = document.createElement('button');
        configButton.className = elements.fullscreenButton.className;
        configButton.classList.add('hidden');
        configButton.id = SCRIPTNAME + '-config-button';
        configButton.innerHTML = core.config.buttonHtml();/*歯車*/
        configButton.setAttribute('title', SCRIPTNAME + '設定');
        configButton.addEventListener('click', core.config.togglePanel, true);
        elements.fullscreenButton.parentNode.insertBefore(configButton, elements.fullscreenButton);
        animate(function(){configButton.classList.remove('hidden')});
      },
      togglePanel: function(){
        if(configPanel) return core.config.closePanel();
        configPanel = document.createElement('div');
        configPanel.id = SCRIPTNAME + '-config-panel';
        configPanel.classList.add('hidden');
        configPanel.innerHTML = core.config.panelHtml();
        configPanel.querySelector('button.cancel').addEventListener('click', core.config.closePanel, true);
        configPanel.querySelector('button.save').addEventListener('click', function(){
          let inputs = configPanel.querySelectorAll('input'), new_configs = {};
          for(let i = 0; inputs[i]; i++) new_configs[inputs[i].name] = inputs[i].value;
          core.config.save(new_configs);
          /* 新しい設定値で再スタイリング */
          core.modify();
          core.addStyle();
          core.scrollComments();
          core.config.closePanel();
        }, true);
        document.body.appendChild(configPanel);
        animate(function(){configPanel.classList.remove('hidden')});
      },
      closePanel: function(){
        configPanel.classList.add('hidden');
        configPanel.addEventListener('transitionend', function(){
          document.body.removeChild(configPanel);
          configPanel = null;
        }, {once: true});
      },
      buttonHtml: function(){
        /* https://www.onlinewebfonts.com/icon/347 */
        return innerHTML = `<!-- iCon by oNlineWebFonts.Com --> <img src="" width="22" height="22">`;
      },
      panelHtml: function(){
        return innerHTML = `
          <h1>${SCRIPTNAME}設定</h1>
          <fieldset>
            <legend>スクロールコメント</legend>
            <p><label>色:                         <input type="color"  name="color"      value="${configs.color}"></label></p>
            <p><label>縁取り色:                   <input type="color"  name="ocolor"     value="${configs.ocolor}"></label></p>
            <p><label>縁取りの太さ(比率):         <input type="number" name="owidth"     value="${configs.owidth}"     min="0" max="0.2" step="0.01"></label></p>
            <p><label>最大行数:                   <input type="number" name="maxlines"   value="${configs.maxlines}"   min="1" max="25"  step="1"></label></p>
            <p><label>行間(比率):                 <input type="number" name="linemargin" value="${configs.linemargin}" min="0" max="1"   step="0.05"></label></p>
            <p><label>不透明度:                   <input type="number" name="opacity"    value="${configs.opacity}"    min="0" max="1"   step="0.05"></label></p>
            <p><label>不透明度(マウスオーバー時): <input type="number" name="hopacity"   value="${configs.hopacity}"   min="0" max="1"   step="0.05"></label></p>
          </fieldset>
          <fieldset>
            <legend>一覧コメント</legend>
            <p><label>文字の不透明度:                   <input type="number" name="lt_opacity"  value="${configs.lt_opacity}"  min="0" max="1" step="0.05"></label></p>
            <p><label>文字の不透明度(マウスオーバー時): <input type="number" name="lt_hopacity" value="${configs.lt_hopacity}" min="0" max="1" step="0.05"></label></p>
            <p><label>背景の不透明度:                   <input type="number" name="lb_opacity"  value="${configs.lb_opacity}"  min="0" max="1" step="0.05"></label></p>
            <p><label>背景の不透明度(マウスオーバー時): <input type="number" name="lb_hopacity" value="${configs.lb_hopacity}" min="0" max="1" step="0.05"></label></p>
          </fieldset>
          <fieldset>
            <legend>アニメーション</legend>
            <p><label>横断にかける秒数: <input type="number" name="duration" value="${configs.duration}" min="1" max="10"  step="1"></label></p>
            <p><label>秒間コマ数:       <input type="number" name="fps"      value="${configs.fps}"      min="1" max="240" step="1"></label></p>
          </fieldset>
          <p class="buttons"><button class="cancel">キャンセル</button><button class="save">保存</button></p>
          <p class="license">Icon made from <a href="http://www.onlinewebfonts.com/icon">Icon Fonts</a> is licensed by CC BY 3.0</p>
        `;
      },
    },
    addStyle: function(){
      if(style) document.head.removeChild(style);
      (function(css){
        style = document.createElement('style');
        style.type = 'text/css';
        style.textContent = css.replace(/^<style>([^]*)<\/style>$/, '$1');
        document.head.appendChild(style);
      })(innerHTML = `<style>
        /* スクロールコメント */
        canvas#${SCRIPTNAME}{
          pointer-events: none;
          position: absolute;
          top: 0;
          left: 0;
          width: 100%;
          height: 100%;
          opacity: ${configs.opacity};
          transition: 500ms ease 0ms;
        }
        body:hover canvas#${SCRIPTNAME}{
          opacity: ${configs.hopacity};
        }
        /* コメントを表示させても映像を画面いっぱいに */
        ${selectors.screen},
        ${selectors.screen} > div{
          width: 100% !important;
          height: 100% !important;
        }
        /* 右コメント一覧を透明に */
        ${selectors.commentPane}{
          mix-blend-mode: hard-light;/*https://stackoverflow.com/questions/15597167/css3-opacity-gradient*/
          background: rgba(0,0,0,${configs.lb_opacity});
          transition: 500ms ease 0ms;
          z-index: 9;/*右側に表示される番組情報や右下のコントローラより下層に*/
        }
        ${selectors.commentPane}:hover{
          background: rgba(0,0,0,${configs.lb_hopacity});
        }
        ${selectors.commentPane}::after{
          pointer-events: none;
          position: absolute;
          content: "";
          left: 0px;
          top: 0px;
          height: 100%;
          width: 100%;
          background: linear-gradient(transparent 50%, gray);
        }
        ${selectors.commentPane} *{
          background: transparent;
          color: rgba(255,255,255,${configs.lt_opacity});
        }
        ${selectors.commentPane}:hover *{
          color: rgba(255,255,255,${configs.lt_hopacity});
        }
        /* 右コメント一覧のスクロールバーを美しく */
        ${selectors.commentPane} > div > div{
          overflow-y: hidden;
        }
        ${selectors.commentPane}:hover > div > div{
          overflow-y: auto;
        }
        ${selectors.commentPane} > div > div::-webkit-scrollbar{
          background: rgba(255,255,255,0);
        }
        ${selectors.commentPane} > div > div::-webkit-scrollbar-thumb{
          background: rgba(255,255,255,${configs.lt_hopacity/2});
        }
        /* マウスオーバー時だけナビゲーションを表示させる */
        body ${selectors.footer}{
          transform: translateY(200%);
        }
        body:hover ${selectors.footer}{
          transform: translateY(0%);
          visibility: visible;
        }
        /* 設定 */
        #${SCRIPTNAME}-config-button{
          right: 125px;
          transition: 500ms ease 0ms;
        }
        #${SCRIPTNAME}-config-button.hidden,
        div[aria-hidden="false"] #${SCRIPTNAME}-config-button/*コメント非表示の時*/{
          bottom: -22px;
        }
        #${SCRIPTNAME}-config-panel{
          position: fixed;
          width: 360px;
          left: 50%;
          bottom: 50%;
          transform: translate(-50%, 50%);
          z-index: 100;
          background: rgba(0,0,0,.75);
          transition: 500ms ease 0ms;
          padding: 5px 0;
        }
        #${SCRIPTNAME}-config-panel.hidden{
          bottom: 0;
          transform: translate(-50%, 100%);
        }
        #${SCRIPTNAME}-config-panel h1,
        #${SCRIPTNAME}-config-panel legend,
        #${SCRIPTNAME}-config-panel p{
          color: rgba(255,255,255,1);
          font-size: 14px;
          padding: 4px 10px;
          line-height:20px;
        }
        #${SCRIPTNAME}-config-panel fieldset p{
          padding-left: 30px;
        }
        #${SCRIPTNAME}-config-panel fieldset p:hover{
          background: rgba(255,255,255,.25);
        }
        #${SCRIPTNAME}-config-panel input{
          width: 80px;
          height: 20px;
          position: absolute;
          right: 10px;
        }
        #${SCRIPTNAME}-config-panel p.buttons{
          text-align: right;
        }
        #${SCRIPTNAME}-config-panel button{
          width: 120px;
          padding: 5px 10px;
          margin-left: 10px;
          border-radius: 5px;
          color: rgba(255,255,255,1);
          background: rgba(64,64,64,1);
          border: 1px solid rgba(255,255,255,1);
        }
        #${SCRIPTNAME}-config-panel button.save{
          font-weight: bold;
          background: rgba(0,0,0,1);
        }
        #${SCRIPTNAME}-config-panel button:hover{
          background: rgba(128,128,128,1);
        }
        #${SCRIPTNAME}-config-panel p.license,
        #${SCRIPTNAME}-config-panel p.license a{
          font-size: 10px;
          color: rgba(255,255,255,.25);
        }
      </style>`);
    },
  };
  let animate = function(callback, ...params){requestAnimationFrame(() => requestAnimationFrame(() => callback(...params)))};
  let innerHTML = '';/*trick for syntax highlighting, waiting js engines support html template*/
  let observe = function(element, callback, config = {childList: true}){
    let observer = new MutationObserver(callback.bind(element));
    observer.observe(element, config);
    return observer;
  };
  let log = (DEBUG) ? function(){
    let l = log.last = log.now || new Date(), n = log.now = new Date();
    console.log(
      SCRIPTNAME + ':',
      /* 00:00:00.000 */ n.toLocaleTimeString() + '.' + n.getTime().toString().slice(-3),
      /* +0.000s      */ '+' + ((n-l)/1000).toFixed(3) + 's',
      /* :00          */ ':' + new Error().stack.match(/:[0-9]+:[0-9]+/g)[1].split(':')[1],/*LINE*/
      /* caller       */ log.caller ? log.caller.name : '',
      ...arguments
    );
    if(arguments.length === 1) return arguments[0];
  } : function(){};
  core.initialize();
  if(window === top) console.timeEnd(SCRIPTNAME);
})();