Greasy Fork is available in English.

ZenzaNicoru

ZenzaWatchを擬似ニコるに対応させるやつ

スクリプトをインストール?
作者が勧める他のスクリプト

ZenzaWatchも気に入るかもしれません。

スクリプトをインストール
質問やレビューの投稿はこちらへ、スクリプトの通報はこちらへどうぞ。
// ==UserScript==
// @name         ZenzaNicoru
// @namespace    https://gitlab.com/u/umany
// @version      0.5.9
// @description  ZenzaWatchを擬似ニコるに対応させるやつ
// @author       umany
// @match          *://www.nicovideo.jp/*
// @match          *://ext.nicovideo.jp/
// @match          *://ext.nicovideo.jp/#*
// @match          *://blog.nicovideo.jp/*
// @match          *://ch.nicovideo.jp/*
// @match          *://com.nicovideo.jp/*
// @match          *://commons.nicovideo.jp/*
// @match          *://dic.nicovideo.jp/*
// @match          *://ex.nicovideo.jp/*
// @match          *://info.nicovideo.jp/*
// @match          *://search.nicovideo.jp/*
// @match          *://uad.nicovideo.jp/*
// @match          *://api.search.nicovideo.jp/*
// @match          *://*.nicovideo.jp/smile*
// @match          *://site.nicovideo.jp/*
// @exclude        *://ads.nicovideo.jp/*
// @exclude        *://www.upload.nicovideo.jp/*
// @exclude        *://www.nicovideo.jp/watch/*?edit=*
// @exclude        *://ch.nicovideo.jp/tool/*
// @exclude        *://flapi.nicovideo.jp/*
// @exclude        *://dic.nicovideo.jp/p/*
// @grant        GM_xmlhttpRequest
// @grant        unsafeWindow
// @connect      j9oi.xyz
// ==/UserScript==

(function () {
  'use strict';

  const load = function () {
    const ZenzaWatch = unsafeWindow.ZenzaWatch;
    const $ = ZenzaWatch.lib.$;

    const gnApiUrl = 'https://j9oi.xyz/api/v1';

    // ニコるくんの画像
    const nicorukun = '';

    let nicoru = {};
    let nicotta = {};
    let watchingId = '';

    // 擬似ニコる受信
    const getNicoru = function (watchId) {
      nicoru = {};
      nicotta = {};
      GM_xmlhttpRequest({
        method: 'GET',
        url: `${gnApiUrl}/getGN.php?a=${watchId}`,
        responseType: 'json',
        onload: (response) => {
          console.log('擬似ニコる受信: %d%O', response.status, response.response);
          nicoru = response.response;
          setNicorareSum();
        }
      });
    };

    // 擬似ニコる送信
    const postNicoru = function (watchId, commentNo) {
      GM_xmlhttpRequest({
        method: 'POST',
        url: `${gnApiUrl}/postGN.php`,
        responseType: 'json',
        data: `m=${watchId}&c=${commentNo}`,
        headers: {
          'Content-Type': 'application/x-www-form-urlencoded'
        },
        onload: (response) => {
          console.log('擬似ニコる送信: %d%O (動画ID: %s, コメント番号: %s)', response.status, response.response, watchId, commentNo);
        }
      });
    };

    // 動画の合計ニコられ数表示
    const setNicorareSum = function () {
      let nicorareSum = 0;
      $.each(nicoru, function (index, value) {
        nicorareSum += value;
      });
      const nicorareSumStr = nicorareSum.toLocaleString ? nicorareSum.toLocaleString() : nicorareSum;

      $('div.VideoMetaInfo.root > span.countOuter').each(function (index) {
        const countOuter = $(this);
        if (countOuter.find('.nicorareCount').length) {
          countOuter.find('.nicorareCount').text(nicorareSumStr);
        } else {
          const column = document.createElement('span');
          $(column).addClass('column');
          const img = document.createElement('img');
          img.src = nicorukun;
          $(img).css({
            width: '18px',
            height: '18px',
            marginTop: '-1px'
          });
          const nicorareCount = document.createElement('span');
          $(nicorareCount).addClass('count nicorareCount').text(nicorareSumStr);
          $(column).append(img, ': ', nicorareCount);
          countOuter.append(column);
        }
      });
    };

    // コメ欄にニコるを設定する
    const setNicoruToCommentList = function (nodeList) {
      $.each(nodeList, function () {
        const commentNo = $(this).data('no');

        /*
         * 通常コメントとコミュニティコメントはクラス'fork0'を持つ
         * 投稿者コメントは'fork1'を持つ
         * 通常コメントとコミュニティコメントも判別したいが厳しいところ
         * 自分のコメントはリロードするまでdata-noが0になるので無視
         */
        if ($(this).hasClass('fork0') && commentNo) {
          setCommentListItemStyle($(this));

          $(this).find('.text').dblclick({
            commentNo: commentNo
          }, executeNicoruHandler);
          $(this).find('.nicoruImage').click({
            commentNo: commentNo
          }, executeNicoruHandler);
        }
      });
    };

    //コメ欄のスタイルを設定
    const setCommentListItemStyle = function (commentListItem) {
      const commentNo = commentListItem.data('no');
      const nicorare = nicoru[commentNo] || 0;

      const text = commentListItem.find('.text');
      const textStyle = {};
      if (nicorare > 0) {
        textStyle.fontWeight = 'bold';
        if (nicorare >= 3 && nicorare < 10) {
          textStyle.color = 'orange';
        } else if (nicorare >= 10) {
          textStyle.color = 'red';
        }
        text.css(textStyle);
      }

      const info = commentListItem.find('.info');
      const nicoruInfoStyle = {
        display: 'inline-block',
        width: '48px'
      };

      const nicoruImageStyle = {
        width: '16px',
        height: '16px',
        float: 'left',
        margin: '2px 5px 2px 0px',
      };

      if (nicotta[commentNo]) {
        nicoruImageStyle.transform = 'rotate(-90deg)';
        nicoruImageStyle.filter = 'grayscale(1)';
      }

      if (info.find('.nicoru').length) {
        nicoruImageStyle.transition = '.5s';
        info.find('.nicoruImage').css(nicoruImageStyle);
        info.find('.nicoruValue').text(nicorare);
      } else {
        const nicoruInfo = document.createElement('span');
        const nicoruImage = document.createElement('img');
        const nicoruValue = document.createElement('span');
        nicoruImage.src = nicorukun;
        nicoruValue.innerText = nicorare;
        $(nicoruImage).addClass('nicoruImage').css(nicoruImageStyle);
        $(nicoruValue).addClass('nicoruValue');
        $(nicoruInfo).addClass('nicoru').css(nicoruInfoStyle).append(nicoruImage, nicoruValue);

        info.prepend(nicoruInfo);
      }
    };

    // ニコる実行ハンドラ
    const executeNicoruHandler = function (e) {
      const commentNo = e.data.commentNo;

      if (!nicotta[commentNo]) {
        postNicoru(watchingId, commentNo);
        nicoru[commentNo] ? nicoru[commentNo]++ : nicoru[commentNo] = 1;
        nicotta[commentNo] = true;

        setCommentListItemStyle($(e.target).parents('.commentListItem'));
        e.preventDefault();
        e.stopPropagation();
      }
    };

    // コメントリスト監視
    const observeCommentList = function (commentList) {
      const observer = new MutationObserver((mutations) => {
        mutations.forEach((mutation) => {
          if (mutation.addedNodes.length > 0) {
            setNicoruToCommentList(mutation.addedNodes);
          }
        });
      });
      const config = {
        childList: true
      };
      observer.observe(commentList.context, config);
    };

    // 動画情報読み込み監視
    const loadVideoInfoHandler = function (data, watchApi, watchId) {
      // console.log('動画情報読み込み (%s : %o)', watchId, data);

      /*
       * so接頭辞の動画ページアクセス時、'watch/soXXXXX'→'watch/[thread_id]'にリダイレクトされる為、
       * watchIdが'soXXXXX'であれば、擬似ニコるAPIに送信する動画IDをthread_idとする
       * (全てのso動画がリダイレクト対象なのか不明瞭なので、また実装変わるかも)
       */
      watchingId = watchId;
      if (watchingId.startsWith('so')) {
        watchingId = data.msgInfo.threadId;
      }
      getNicoru(watchingId);
    };

    ZenzaWatch.emitter.on('loadVideoInfo', loadVideoInfoHandler);
    Object.defineProperty(ZenzaWatch.debug, '$commentList', {
      get: function () {
        return this._$commentList;
      },
      set: function (commentList) {
        this._$commentList = commentList;
        observeCommentList(commentList);
      }
    });
  };

  // ZenzaWatch待機
  const waitForZenzaWatch = function () {
    if (unsafeWindow.ZenzaWatch && unsafeWindow.ZenzaWatch.ready) {
      console.log('ZenzaWatch is Ready');
      load();
    } else {
      document.body.addEventListener('ZenzaWatchInitialize', () => {
        console.log('onZenzaWatchInitialize');
        load();
      });
    }
  };

  waitForZenzaWatch();
})();