48CO Mod

Google+(ぐぐたす)で AKB48 グループメンバーのコメントを抽出するユーザースクリプト

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name        48CO Mod
// @namespace   http://gplus.to/fronoske/
// @version     2.1.0
//
// @description Google+(ぐぐたす)で AKB48 グループメンバーのコメントを抽出するユーザースクリプト
// @icon        https://s3.amazonaws.com/uso_ss/icon/120097/large.png
// @author      originally Ming-Hsien Lin (akiratw), revised by @fronoske
// @license     MIT License
//
//
// @match       https://plus.google.com/*
// @include     https://plus.google.com/*
// @exclude     https://plus.google.com/*_/*
//
// @require     https://cdnjs.cloudflare.com/ajax/libs/jquery/2.1.3/jquery.min.js
//
// @grant       GM_addStyle
// @grant       GM_setClipboard
//
// @run-at      document-idle
// @noframes
// ==/UserScript==

/*
■変更履歴  
2015.02.14 メンバーデータの取得日時の表示を改善
2015.12.07 新レイアウトに対応
           レイアウトの新旧によって各所の処理を分岐
           href="#" だとサイト側の javascript がイベントに反応するため href="javascript:void(0)" に変更
2015.12.13 コードを大幅に追加
           抽出コメントの「+nn」をクリックすることで+1したユーザーを表示するようにした(新旧)
           コメント欄のクリックで意図せずスクロールしていたのを抑止するようにした(新レイアウト)
           自分の抽出コメントを編集したり削除したりしたときにそれが反映されなかった問題を修正(新旧)

■制限事項
 - コメント抽出タイミングは「カーソルを合わせたとき」固定
 - コメントリストの位置は「投稿内容の下」固定
*/

var CommentsOnly = function () {
  var self = this;

  self.NAME = '48 Comments Only';
  self.VERSION = '2.0.0';
  self.IS_CHROME = !!window.chrome && !!window.chrome.runtime;
  self.SPREADSHEET_KEY = '1uWMxVR0MjwL2p-N2k0WuG096qIDtSA92tsTSjKV3w_8';
  self.SPREADSHEET_URL = 'https://docs.google.com/spreadsheets/d/' + self.SPREADSHEET_KEY;

  this.SETTINGS_KEY = 'userscript.48comod';

  self.OLD_LAYOUT = !$("body").attr("jscontroller"); // 2015年秋のレイアウト変更前かどうか
  self.INDIVIDUAL_POST = ( $("div[data-iid]").length == 0 );
  // console.log( $("body").attr("jscontroller")   );
  // console.log("[48CO] old_layout="+self.OLD_LAYOUT);
  // console.log("[48CO] individual?=" + self.INDIVIDUAL_POST);
  
  if (self.OLD_LAYOUT){
    this.SELECTORS = {
    banner: '#gbqf',
    post: 'div[id^="update-"]',
    author: 'header a[oid]',
    content: 'div.Al.pf',
    comments: 'div.KK.gR',
    sharers: 'div.eK',
    plusone: {
      button: '.esw',
      pressed: '.eswa',
      normal: '.eswd'
      }
    };
  }else if ( self.INDIVIDUAL_POST ){ // 個別の投稿ページ
    this.SELECTORS = {
    banner: 'div.Vrm0oe',
    post: 'div[role="main"]', // 1つの投稿全体
    author: 'div[id^="author:"] a[href]',
    content: 'div.KwDIr',
    comments: 'div.pQGRZe',
    sharers: 'div.eK',
    plusone: {
      button: '.esw',
      pressed: '.eswa',
      normal: '.eswd'
      }
    };
  }else{ // ストリーム
    this.SELECTORS = {
    banner: 'div.Vrm0oe', //バナー全体
    post: 'div[data-iid]', // 1つの投稿全体
    author: 'div[id^="author:"] a[href]', // 
    content: 'div.KwDIr',
    comments: 'div.pQGRZe',
    sharers: 'div.eK',
    plusone: {
      button: '.esw',
      pressed: '.eswa',
      normal: '.eswd'
      }
    };
  }

  this.ICONS = {
    loading: '',
    icon: '',
    settings: '',
    collapse: '',
    expand: '',
    dropdown: '',
    block: '',
    blocked: '',
    addUser: '',
    close: '',
    copy: ''
  };

  this.API = new GooglePlusAPI();
  // 2015.12.13 @fronoske セッションを早めに取得しておく。新レイアウトの場合はこの1回だけ同期的にajax呼び出し。
  this.API._getSession({async: false});

  this._settings = {
    gplusIDs: [],
    blockIDs: [],
    commentLoadTrigger: 'mouseenter',
    commentLoadInterval: 30000,
    loadSharers: true,
    highlightMyComments: true,
    commentsPosition: 'content',
    commentStyle: 'normal',
    triggerDelay: 0,
    autoExpand: true,
    hideStreamComments: false,
    displayCommentNumber: true,
    displayCommentHumanTime: true,
    colors: ['#FFF2F6', '#FFE6EE', '#FF3377'],
    memberData: {}
  };

  this._members = {
    groups: {},
    members: {},
    updated: null
  };

  this._login_user = {
    id: null,
    name: null,
    photo: null
  };

  this._idRegex = null;

  this._lang = {};

  this._readSettings();

  this._initData(function () {
    self._localize();
    self._renderGeneralUI();
    self._renderCommentUI();
    self._renderSettingsUI();
    self._writeSettings();
  });

  return this;
};

CommentsOnly.prototype._renderGeneralUI = function () {
  var self = this;

  /**
   * Adds CSS.
   */
  var addStyle = function () {
    self._addStyle('general',
      '.ext48co { color: #333; font: normal 12px Roboto, Arial, sans-serif; }' +
      '.ext48co a { color: #36C; text-decoration: none; }' +
      '.ext48co a:hover { text-decoration: underline; }' +
      'button.ext48co { cursor: pointer; margin: 2px 5px; padding: 5px 15px; border: 1px solid #CCC; border-radius: 2px; background: #FAFAFA; color: #333; font-weight: bold; }' +
      'button.ext48co:disabled { cursor: default; opacity: 0.5; }' +
      'button.ext48co:hover { box-shadow: 0 1px 1px #CCC; }' +
      'button.ext48co.ext48co-submit { border-color: #29691D; background: #3D9400; color: #FFF; }' +
      'button.ext48co.ext48co-main { border-color: #3079ED; background: #4D90FE; color: #FFF; }' +
      'button.ext48co.ext48co-extButton { margin: 0 2px; border-color: 1px solid rgba(0, 0, 0, 0.2); background: #CC295F; color: #FFF; font-size: 11px; }' +
      'input.ext48co { vertical-align: middle; }' +
      'input[type="text"].ext48co { margin: 2px; padding: 5px; font: normal 12px Roboto, Arial, sans-serif; }');
  };

  addStyle();
};

CommentsOnly.prototype._renderCommentUI = function () {
  var self = this;

  /**
   * Adds CSS.
   */
  var addStyle = function () {
    self._addStyle('comments',
      'div.ext48co-comments { display: none; padding: 5px; background: ' + self._settings.colors[1] + '; }' +
      'div.ext48co-comments.ext48co-popup { z-index: 9999; position: absolute; left: 480px; top: 30px; width: 360px; border-radius: 4px; box-shadow: 0 0 5px #CCC; }' +
      'div.ext48co-comments.ext48co-popup:after { content: ""; position: absolute; top: 10px; left: -10px; border-width: 10px 10px 10px 0; border-style: solid; border-color: transparent ' + self._settings.colors[1] + '; }' + self.SELECTORS.content + ' div.ext48co-comments { position: relative; margin: 20px 0 10px 0; border-radius: 4px; }' + self.SELECTORS.content + ' div.ext48co-comments:after { content: ""; position: absolute; top: -10px; left: 10px; border-width: 0 10px 10px; border-style: solid; border-color: ' + self._settings.colors[1] + ' transparent; }' +
      'div.ext48co-status { overflow: hidden; height: 18px; padding: 5px 50px 5px 5px; color: ' + self._settings.colors[2] + '; text-overflow: ellipsis; white-space: nowrap; }' +
      'div.ext48co-status img { margin: 0 2px; opacity: 0.5; }' +
      'div.ext48co-status a:hover img { opacity: 1; }' +
      'span.ext48co-status-text { cursor: pointer; }' +
      'img.ext48co-loading { padding: 0 5px 0 0; vertical-align: middle; }' +
      'span.ext48co-status-buttons { position: absolute; right: 5px; }' +
      'div.ext48co-comment { padding: 5px 5px 8px 5px; }' +
      'div.ext48co-comment:hover div.ext48co-comment-number { visibility: visible; }' +
      'div.ext48co-comment:hover span.ext48co-comment-plusone button { visibility: visible; }' +
      'div.ext48co-comment-myself { background: #FAFAFA; }' +
      'div.ext48co-comment-member { background: ' + self._settings.colors[0] + '; }' +
      'img.ext48co-comment-photo { position: absolute; border-radius: 2px; }' +
      'div.ext48co-comment-number { visibility: hidden; position: absolute; right: 5px; color: #CCC; font-size: 10px; }' +
      'div.ext48co-comment-content { position: relative; margin-left: 55px; }' +
      'div.ext48co-comment-author { font-weight: bold; }' +
      'div.ext48co-comment-body { margin: 2px 0; }' +
      'div.ext48co-comment-time { color: #999; }' +
      'a.ext48co-copy { opacity: 0.5; }' +
      'a.ext48co-copy:hover { opacity: 1; }' +
      'span.ext48co-comment-plusone { position: absolute; right: 5px; }' +
      // 2015.12.14 @fronoske +1ユーザーダイアログ
      'div.ext48co-po-userlist-dialog { width: 200px; padding: 5px 10px; z-index: 1200; }' +
      'div.ext48co-po-userlist-dialog div.header { float: left; width:160px; margin: 1em auto; }' +
      'div.ext48co-po-userlist-dialog div.close { padding: 4px; width:auto; float:right; text-align:right; font-size: normal; color: gray; cursor: pointer }' +
      'div.ext48co-po-userlist-dialog div.userlist { margin: 5px; clear: both; width: 100%; max-height: 300px; overflow-x: hidden; overflow-y: auto; text-overflow: ellipsis; }' +
      'div.ext48co-po-userlist-dialog div.userlist div.user { margin: 0.3em auto; }' +
      'div.ext48co-po-userlist-dialog div.userlist div.user * { vertical-align: middle; text-decoration: none; }' +
      // 2015.12.14 @fronoske 抽出コメントの「+nn」のカーソルもpointerに
      'span.ext48co-comment-plusone { cursor: pointer; }' +
      'span.ext48co-comment-plusone button { visibility: hidden; vertical-align: top; cursor: pointer; border: 1px solid #D9D9D9; border-radius: 2px; background: #FFF; color: #262626; font: bold 11px Roboto, Arial, sans-serif; }' +
      'span.ext48co-comment-plusone button.ext48co-comment-plused { visibility: visible; border-color: #FFF; background: #DD4B39; color: #FFF; }' +
      'div.ext48co-comment-ballon { position: relative; padding: 5px; border: 1px solid #CCC; border-radius: 4px; background: #FFF; }' +
      'div.ext48co-comment-ballon:after { content: ""; position: absolute; top: 15px; left: -5px; border-width: 5px 5px 5px 0; border-style: solid; border-color: transparent #FFF; }' +
      'div.ext48co-comment-ballon:before { content: ""; position: absolute; top: 13px; left: -7px; border-width: 7px 7px 7px 0; border-style: solid; border-color: transparent #CCC; }' +
      'div.ext48co-comment-editor textarea { width: 90%; height: 50px; padding: 5px; border: 1px solid #CCC; font-size: 12px; }');
  };

  /**
   * Bind event handler functions.
   */
  var bindEvents = function () {
    $(document).on('loadComments', self.SELECTORS.post, loadComments);
    $(document).on('initComments', self.SELECTORS.post, initComments);
    $(document).on(self._settings.commentLoadTrigger, self.SELECTORS.post, function () {
      // console.log("[48CO] triggered!")
      var $this = $(this);
      if (self._settings.triggerDelay) {
        setTimeout(function () {
          $this.trigger('loadComments');
        }, self._settings.triggerDelay);
      } else {
        $this.trigger('loadComments');
      }
    });

    $(document).on('toggleComments', 'div.ext48co-comments', toggleComments);
    $(document).on('addComments', 'div.ext48co-commentlist', addComments);
    $(document).on('addSharers', 'div.ext48co-sharerlist', addSharers);
    $(document).on('plusOneComment', 'div.ext48co-comment', plusOneComment);
    $(document).on('openEditor', 'div.ext48co-comment', openEditor);
    $(document).on('closeEditor', 'div.ext48co-comment', closeEditor);
    $(document).on('saveComment', 'div.ext48co-comment', saveComment);
    $(document).on('deleteComment', 'div.ext48co-comment', deleteComment);
    $(document).on('copyComment', 'div.ext48co-comment', copyComment);
    $(document).on('setStatusText', 'div.ext48co-status', setStatusText);

    $(document).on('click', 'div.ext48co-status', function () {
      var $comments = $(this).parents('div.ext48co-comments');
      $comments.trigger('toggleComments');
    });

    $(document).on('click', 'span.ext48co-comment-plusone button', function (event) {
      var $comment = $(this).parents('div.ext48co-comment');
      $comment.trigger('plusOneComment');
    });

    $(document).on('click', 'a.ext48co-comment-edit', function (event) {
      event.preventDefault();
      var $comment = $(this).parents('div.ext48co-comment');
      $comment.trigger('openEditor');
    });

    $(document).on('click', 'button.ext48co-comment-cancel', function (event) {
      var $comment = $(this).parents('div.ext48co-comment');
      $comment.trigger('closeEditor');
    });

    $(document).on('click', 'button.ext48co-comment-save', function (event) {
      var $comment = $(this).parents('div.ext48co-comment');
      $comment.trigger('saveComment');
    });

    $(document).on('click', 'button.ext48co-comment-delete', function (event) {
      var $comment = $(this).parents('div.ext48co-comment');
      $comment.trigger('deleteComment');
    });

    $(document).on('click', 'a.ext48co-button-toggle', function (event) {
      event.preventDefault();
    });

    $(document).on('click', 'a.ext48co-copy', function (event) {
      event.preventDefault();
      var $comment = $(this).parents('div.ext48co-comment');
      $comment.trigger('copyComment');
    });

    // 2015.12.14 @fronoske
    $(document).on('plusOneUserList', 'div.ext48co-comment', plusOneUserList);
    $(document).on('click', 'span.ext48co-comment-plusone', function (event) {
      if (event.target.localName == 'button') return;
      var $comment = $(this).parents('div.ext48co-comment');
      $comment.trigger('plusOneUserList');
    });
  };

  /**
   * Initializes comments list and attaches post data to element.
   */
  var initComments = function () {
    var $post = $(this);
    // 2015.12.07 @fronoske 新旧レイアウトによって authorID と postID の取得方法を変える
    if (self.OLD_LAYOUT) {
      var authorID = $post.find(self.SELECTORS.author).eq(0).attr('oid');
      var postID = $post.prop('id').match(/^update-(.+)$/)[1];
    } else {
      var authorID = $post.find(self.SELECTORS.author).eq(0).attr('href').replace("./", "");
      var postID = $post.attr('data-iid') || $post.find("div[jscontroller]").eq(0).attr('jsdata').replace(/^\w+;|;\w+$/g, "");
      // console.log("authorID=" + authorID);
      // console.log( $post.find("div[jscontroller]").eq(0).attr('jsdata')  );
      // console.log("postID=" + postID);
    }

    if (!authorID.match(self._idRegex)) {
      $post.data('ext48co.disabled', true);
      return;
    }

    $post.data('ext48co.id', postID);
    $post.data('ext48co.timestamp', 0);
    $post.data('ext48co.locked', false);

    var html =
      // 2015.12.13 @fronoske スクロールされるのを抑止するためにjsaction属性を追加
      '<div class="ext48co ext48co-comments" jsaction="click:null">' +
      '<div class="ext48co-status">' +
      '<img class="ext48co-loading" src="' + self.ICONS.loading + '">' +
      '<span class="ext48co-status-text"></span>' +
      '<span class="ext48co-status-buttons">' +
      '<a href="#" class="ext48co-button-settings" title="' + self._('48 Comments Only Settings') + '"><img src="' + self.ICONS.settings + '"></a>' +
      '<a href="#" class="ext48co-button-toggle" title="' + self._('Collapse or expand comments') + '"><img src="' + (self._settings.autoExpand ? self.ICONS.collapse : self.ICONS.expand) + '"></a>' +
      '</span>' +
      '</div>' +
      '<div class="ext48co-commentlist"' + (self._settings.autoExpand ? '' : ' style="display: none"') + '></div>' +
      '<div class="ext48co-sharerlist"></div>' +
      '</div>';

    if (self.OLD_LAYOUT){
      if (self._settings.commentsPosition == 'content') {
        $post.find(self.SELECTORS.content).append(html);
        $post.find('div.ext48co-comments').fadeIn(200);
      } else if (self._settings.commentsPosition == 'list') {
        $post.find(self.SELECTORS.comments).parent().before(html);
        $post.find('div.ext48co-comments').fadeIn(200);
      } else if (self._settings.commentsPosition == 'popup') {
        $post.find(self.SELECTORS.content).after(html);
        $post.find('div.ext48co-comments').addClass('ext48co-popup').fadeIn(200);
      }
    } else {
      if (self._settings.commentsPosition == 'content' || self._settings.commentsPosition == 'list' ) {
        $post.find(self.SELECTORS.content).append(html);
        $post.find('div.ext48co-comments').fadeIn(200);
      } else if (self._settings.commentsPosition == 'popup') {
        $post.find(self.SELECTORS.content).after(html);
        $post.find('div.ext48co-comments').addClass('ext48co-popup').fadeIn(200);
      }
    }

    if (self._settings.hideStreamComments && location.href.indexOf('/posts/') == -1) {
      $post.find(self.SELECTORS.comments).fadeOut(200);
    }
  };

  /**
   * Loads comments.
   */
  var loadComments = function (event) {
    var $post = $(this);
    var now = new Date();

    if (!$post.data('ext48co.id')) $post.trigger('initComments');

    if ($post.data('ext48co.disabled')) return;
    if (now.getTime() - $post.data('ext48co.timestamp') < self._settings.commentLoadInterval) return;
    if ($post.data('ext48co.locked')) return;

    $post.data('ext48co.timestamp', now.getTime());
    $post.data('ext48co.locked', true);

    var $status = $post.find('div.ext48co-status');
    var $commentList = $post.find('div.ext48co-commentlist');
    var $sharerList = $post.find('div.ext48co-sharerlist');

    $status.trigger('setStatusText', [self._('Loading...')]);

    // Send AJAX request to fetch all comments and parse them.
    self.API.lookupPost($post.data('ext48co.id'), function (response) {
      if (response.success) {
        var comments = response.data.comments;
        $commentList.trigger('addComments', [comments]);
      } else {
        var errorStatus = self._('HTTP % - %', [response.data.error, response.data.text]);
        $status.trigger('setStatusText', [self._('Error occurred.'), errorStatus]);
      }

      now = new Date();
      $post.data('ext48co.timestamp', now.getTime());
      $post.data('ext48co.locked', false);
    });

    if ($post.find(self.SELECTORS.sharers).length !== 0 && self._settings.loadSharers) {
      // Send AJAX request to fetch all sharers and parse them.
      self.API.lookupSharers($post.data('ext48co.id'), function (response) {
        if (response.success) {
          var sharers = response.data;
          $sharerList.trigger('addSharers', [sharers]);
        } else {
          var errorStatus = self._('HTTP % - %', [response.data.error, response.data.text]);
          $status.trigger('setStatusText', [self._('Error occurred.'), errorStatus]);
        }
      });
    }
  };

  /**
   * Adds comments to list.
   */
  var addComments = function (event, comments) {
    var $commentList = $(this);
    var $post = $commentList.parents(self.SELECTORS.post);
    var $status = $commentList.prev();
    var memberComments = [];
    var myComments = [];
    var names = [];
    var latest = null;

    comments.forEach(function (element, index) {
      var postBy;

      if (element.author.id == self._login_user.id && self._settings.highlightMyComments) {
        postBy = 'myself';
      } else if (element.author.id.match(self._idRegex)) {
        postBy = 'member';
      } else {
        return;
      }

      var commentTime = element.edited || element.time;
      var commentHTML =
        '<div class="ext48co-comment ext48co-comment-' + postBy + '" data-id="' + element.id + '" data-authorid="' + element.author.id +
        // 2015.12.14 @fronoske plusoneIdを持たせた
        '" data-plusoneid="' + element.plusone.id + '">' +
        '<a href="./' + element.author.id + '" oid="' + element.author.id + '"><img class="ext48co-comment-photo" src="' + element.author.photo + '?sz=48"></a>' +
        '<div class="ext48co-comment-content' + (self._settings.commentStyle == 'ballon' ? ' ext48co-comment-ballon' : '') + '">' +
        '<div class="ext48co-comment-number">' +
        '<!--<a href="javascript:void(0)" class="ext48co-copy">' + self._('Copy') + '</a>--> ' + (self._settings.displayCommentNumber ? (index + 1) : '') +
        '</div>' +
        '<div class="ext48co-comment-author">' +
        '<a href="./' + element.author.id + '" oid="' + element.author.id + '">' + element.author.name + '</a>' +
        ' ' + ((self._settings.gplusIDs.indexOf(element.author.id) !== -1) ? '<small>(' + self._('Custom') + ')</small>' : '') +
        '</div>' +
        '<div class="ext48co-comment-body">' + element.content + '</div>' +
        '<div class="ext48co-comment-time">' +
        '<span title="' + commentTime.toLongString() + '">' + (self._settings.displayCommentHumanTime ? commentTime.toHumanTimeDiff(self._('humantime')) + ' - ' : '') + commentTime.toShortString() + (element.edited ? self._(' (edited)') : '') +
        '</span>' + (postBy == 'myself' ? ' <a href="javascript:void(0)" class="ext48co-comment-edit">' + self._('Edit') + '</a>' : '') + (self._login_user.id ? '<span class="ext48co-comment-plusone">' +
        '<button id="po-' + element.id + '"' +
        ' class="' + (element.plusone.isPlused ? 'ext48co-comment-plused' : '') + '"' +
        ' g:entity="comment:' + element.id + '"' +
        ' g:token="true">+1' +
        '</button>' + (element.plusone.count ? ' +<strong>' + element.plusone.count + '</strong>' : '') +
        '</span>' : '') +
        '</div>' +
        '</div>' +
        '</div>';

      if (postBy == 'member') {
        memberComments.push(commentHTML);
        latest = commentTime;
        if (names.indexOf(element.author.name) == -1) names.push(element.author.name);
        $post.find('div[id="' + element.id + '"]').addClass('ext48co-comment-member');
      } else {
        myComments.push(commentHTML);
        $post.find('div[id="' + element.id + '"]').addClass('ext48co-comment-myself');
      }
    });

    $commentList.empty().html(memberComments.join(' ') + myComments.join(' '));

    var commentCount = memberComments.length;
    var name = names.reverse().join(self._(', '));
    if (commentCount) {
      var statusText = self._('% comments', commentCount) + ' - ' + latest.toHumanTimeDiff(self._('humantime')) + ' - ' + name;
      $status.trigger('setStatusText', [statusText, name]);
    } else {
      $status.trigger('setStatusText', [self._('No comment from members')]);
    }
  };

  /**
   * Add sharers to list.
   */
  var addSharers = function (event, sharers) {
    var $sharerList = $(this);
    var output = [];

    sharers.forEach(function (element, index) {
      if (!element.id.match(self._idRegex)) return;
      output.push(
        '<img src="' + element.photo + '?sz=24" style="vertical-align: middle"> ' +
        '<a href="./' + element.id + '" oid="' + element.id + '">' + element.name + '</a>');
    });

    if (output.length) {
      $sharerList.html('<div class="ext48co-comment ext48co-comment-member"> ' + self._('% shared this post.', output.join(self._(', '))) + '</div>');
    } else {
      $sharerList.empty();
    }
  };

  /**
   * +1 a comment.
   */
  var plusOneComment = function (event) {
    var $comment = $(this);
    var $post = $comment.parents(self.SELECTORS.post);
    var $button = $comment.find('span.ext48co-comment-plusone button');
    var $counter = $comment.find('span.ext48co-comment-plusone strong');
    var count = parseInt($counter.text(), 10);
    var now = new Date();

    $post.data('ext48co.timestamp', now.getTime());

    if ($counter.length === 0) {
      $button.after('&nbsp; +<strong>1</strong>');
    }

    if ($button.hasClass('ext48co-comment-plused')) {
      $button.prop('disabled', true);
      $button.removeClass('ext48co-comment-plused');
      self.API.plusOneComment($comment.data('id'), false, function (response) {
        count -= 1;
        $counter.text(count);
        $button.prop('disabled', false);
      });
    } else {
      $button.prop('disabled', true);
      $button.addClass('ext48co-comment-plused');
      self.API.plusOneComment($comment.data('id'), true, function (response) {
        count += 1;
        $counter.text(count);
        $button.prop('disabled', false);
      });
    }
  };

  /**
   * +1 user list (2015.12.14 @fronoske)
   */
  var plusOneUserList = function (event) {
    var $comment = $(this);
    var $counter = $comment.find('span.ext48co-comment-plusone strong');
    var count = parseInt($counter.text(), 10);

    // console.log("[48CO] call plusOneUserList");
    self.API.plusOneUserList($comment.data('plusoneid'), function (response) {
      var userList = response.data[1];
      var plusOneUsers = [];
      userList.forEach(function (user, index) {
       // console.dir(user);
        plusOneUsers.push( {
          name: user[0],
          id: user[1],
          photo: user[3],
          altName: user[4]
          });
      });
      var $dialog = $('<div class="ext48co ext48co-po-userlist-dialog"></div>');
      var html = '<div class="header"><strong>' + self._("% people +1'd this comment", count) + '</strong></div>' +
        '<div class="close">&#10005;</div>' +
          '<div class="userlist">' +
        $.map(plusOneUsers, function(user, index){
          return '<div class="user"><a href="./' + user.id + '"><img src="' + user.photo + '?sz=28">&nbsp;&nbsp;<span>' + user.name + '</span></a></div>'
        }).join("\n") +  '</div>';
      $dialog.html(html);
      // console.log("[48CO] plusOneUserList HTML= " + html);
      $('body').append($dialog);
      $dialog.css({
      'position': 'absolute',
      'top': $counter.offset().top,
      'left': $counter.offset().left,
      'opacity': 1.0,
      'background': 'white',
      'box-shadow': '5px 2px 2px black'
      });

      /*
      var centerY = $(window).innerHeight() / 2;
      var elementTop = $counter.offset().top - $(window).scrollTop();
      if ( elementTop > centerX ){
        $dialog.offset( {top: $counter.offset().top - $dialog.height()});
      }
      */
      $dialog.fadeIn(500);
      $(document).one('click', function(event) {
        if(!$.contains($dialog, event.target)){
          // console.log("[48CO] click to close");
          $dialog.fadeOut(500, function(){
            $dialog.remove();
          });
        }
      });
    });
  };

  /**
   * Opens comment editor.
   */
  var openEditor = function (event) {
    var $comment = $(this);
    var $post = $comment.parents(self.SELECTORS.post);
    var $commentBody = $comment.find('div.ext48co-comment-body');

    if ($comment.find('div.ext48co-comment-editor').length !== 0) return;

    $post.data('ext48co.locked', true);

    var comment = $commentBody.html()
      .replace(/<br>/g, "\n")
      .replace(/(<b>|<\/b>)/g, '*')
      .replace(/(<i>|<\/i>)/g, '_')
      .replace(/(<s>|<\/s>)/g, '-')
      .replace(/(<a\s([^>]+)>|<\/a>)/g, '')
      .replace(/(<span\s([^>]+)>|<\/span>)/g, '');

    $commentBody.hide()
      .after(
      '<div class="ext48co-comment-editor">' +
      '<textarea>' + comment + '</textarea><br>' +
      '<button class="ext48co ext48co-submit ext48co-comment-save">' + self._('Save') + '</button>' +
      '<button class="ext48co ext48co-comment-delete">' + self._('Delete') + '</button>' +
      '<button class="ext48co ext48co-comment-cancel">' + self._('Cancel') + '</button>' +
      '</div>');

    $comment.find('div.ext48co-comment-editor textarea').focus();
  };

  /**
   * Closes comment editor.
   */
  var closeEditor = function (event) {
    var $comment = $(this);
    var $post = $comment.parents(self.SELECTORS.post);
    var $commentBody = $comment.find('div.ext48co-comment-body');

    $post.data('ext48co.locked', false);
    $commentBody.show();
    $comment.find('div.ext48co-comment-editor').remove();
  };

  /**
   * Saves edited comment.
   */
  var saveComment = function (event) {
    var $comment = $(this);
    var $editor = $comment.find('div.ext48co-comment-editor');
    var $textarea = $editor.find('textarea');
    var $commentBody = $comment.find('div.ext48co-comment-body');

    $editor.find('textarea, button').prop('disabled', true);

    self.API.editComment($comment.data('id'), $textarea.val(), function (response) {
      if (response.success) {
        $commentBody.html(response.data.content);
      }
      $comment.trigger('closeEditor');
    });
  };

  /**
   * Deletes the comment.
   */
  var deleteComment = function (event) {
    var $comment = $(this);
    var answer = confirm(self._('Do you want to permanently delete this comment?'));

    if (answer) {
      self.API.deleteComment($comment.data('id'), function (response) {
        if (response.success) {
          $comment.fadeOut(200, function () {
            $comment.trigger('closeEditor').remove();
          });
        }
      });
    }
  };

  /**
   * Collapses or expands comment list.
   */
  var toggleComments = function (event) {
    var $this = $(this);
    var $commentList = $this.find('div.ext48co-commentlist');
    var $buttonImage = $this.find('a.ext48co-button-toggle img');

    if ($commentList.is(':visible')) {
      $commentList.fadeOut(200);
      $buttonImage.prop('src', self.ICONS.expand);
    } else {
      $commentList.fadeIn(200);
      $buttonImage.prop('src', self.ICONS.collapse);
    }
  };

  var copyComment = function (event) {
    var $comment = $(this);
    var comment = $comment.find('.ext48co-comment-body').html()
      .replace(/<br>/g, "\n")
      .replace(/(<b>|<\/b>)/g, '')
      .replace(/(<i>|<\/i>)/g, '')
      .replace(/(<s>|<\/s>)/g, '')
      .replace(/(<a\s([^>]+)>|<\/a>)/g, '')
      .replace(/(<span\s([^>]+)>|<\/span>)/g, '');

    var data = $comment.find('.ext48co-comment-author').text() + ":\n" + comment + "\n" +
      "(" + $comment.find('.ext48co-comment-time span').eq(0).text() + ")\n";

    GM_setClipboard(data);
  };

  /**
   * Sets comments status bar text.
   */
  var setStatusText = function (event, text, title) {
    var $this = $(this);
    var $status = $this.find('span.ext48co-status-text');
    var $loading = $this.find('img.ext48co-loading');

    if (text == self._('Loading...')) {
      $loading.show();
    } else {
      $loading.hide();
    }

    $status.html(text)
      .prop('title', title);
  };

  addStyle();
  bindEvents();
};

CommentsOnly.prototype._renderSettingsUI = function () {
  var inFrame = (top != self);
  var self = this;

  /**
   * Adds CSS.
   */
  var addStyle = function () {
    self._addStyle('settings', (inFrame ? '' : ' a.ext48co-button-settings { display: none; }') +
      '#ext48co-settings-button { display: inline-block; vertical-align: top; }' +
      '#ext48co-settings-dialog { display: none; position: absolute; z-index: 988; top: 0; left: 0; width: 50%; min-width: 720px; padding: 50px 10px 10px 10px; border: 1px solid #CCC; box-shadow: 0 0 20px #CCC; background: #FFF; }' +
      '#ext48co-settings-dialog h1 { margin: 0; color: #CC295F; font-size: 18px; font-weight: normal; }' +
      '#ext48co-settings-dialog p { margin: 5px 0; }' +
      '#ext48co-settings-dialog label { display: inline-block; min-width: 200px; }' +
      '#ext48co-settings-close { position: absolute; top: 60px; right: 20px; }' +
      '#ext48co-settings-main { overflow: auto; min-height: 300px; margin: 10px 0 60px 0; }' +
      '#ext48co-settings-menu { position: absolute; width: 160px; max-width: 160px; border-right: 1px solid #E5E5E5; }' +
      '#ext48co-settings-menu a { display: block; padding: 10px; border-left: 3px solid transparent; color: #333; }' +
      '#ext48co-settings-menu a.ext48co-selected { border-color: #CC295F; color: #CC295F; }' +
      '#ext48co-settings-menu a:hover { background: #FAFAFA; text-decoration: none; }' +
      '.ext48co-settings-content { position: relative; margin-left: 180px; }' +
      '#ext48co-settings-info { position: absolute; bottom: 10px; left: 10px; color: #999; font-size: 11px; }' +
      '#ext48co-settings-buttons { position: absolute; bottom: 10px; right: 10px; }' +
      '#ext48co-settings-groups { padding: 5px; border-top: 1px solid #E5E5E5; background: #FAFAFA; }' +
      '#ext48co-settings-groups a { padding: 5px 10px; color: #333; font-weight: bold; text-decoration: none; }' +
      '#ext48co-settings-groups a:hover { color: #CC295F; }' +
      '#ext48co-settings-groups a.ext48co-selected { background: #CC295F; color: #FFF; }' +
      '#ext48co-settings-teams { padding: 8px 5px 5px 5px; background: #CC295F; }' +
      '#ext48co-settings-teams a { margin: 0 2px; padding: 5px 10px; color: #FFF; text-decoration: none; }' +
      '#ext48co-settings-teams a.ext48co-selected { background: #FFF; color: #CC295F; }' +
      '#ext48co-settings-users { overflow: auto; margin: 0; padding: 0; height: 300px; max-height: 300px; list-style: none; }' +
      '#ext48co-settings-users li { overflow: hidden; padding: 5px 0; border-bottom: 1px dotted #CCC; text-overflow: ellipsis; white-space: nowrap; }' +
      '#ext48co-settings-users li a { margin: 0 5px; font-weight: bold; }' +
      '#ext48co-settings-users li span { color: #999; font-size: 11px; }' +
      '#ext48co-settings-users li img { vertical-align: middle; }' +
      '#ext48co-settings-users li img.ext48co-settings-photo { width: 24px; height: 24px; }' +
      'img.ext48co-settings-photo { width: 24px; height: 24px; background: #E9ECEE; vertical-align: middle; }' +
      'a.ext48co-settings-userbutton { opacity: 0.5; }' +
      'a.ext48co-settings-userbutton:hover { opacity: 1; }' +
      '.ext48co-desc { color: #999; }');
  };

  /**
   * Inserts buttons.
   */
  var initButton = function () {
    $(self.SELECTORS.banner)
      .append(
      '<div id="ext48co-settings-button">' +
      '  <button class="ext48co ext48co-extButton ext48co-button-settings" title="' + self._('48 Comments Only Settings') + '">' + self.NAME + '</button>' +
      '</div>');
  };

  /**
   * Binds event handler functions.
   */
  var bindEvents = function () {
    $(document).on('toggleDialog', '#ext48co-settings-dialog', toggleDialog);
    $(document).on('readSettings', '#ext48co-settings-dialog', readSettings);
    $(document).on('saveSettings', '#ext48co-settings-dialog', saveSettings);
    $(document).on('resetSettings', '#ext48co-settings-dialog', resetSettings);
    $(document).on('checkUpdate', '#ext48co-update', checkUpdate);

    $(document).on('addUser', '#ext48co-settings-users li', addUser);
    $(document).on('editUser', '#ext48co-settings-users li', editUser);
    $(document).on('blockMember', '#ext48co-settings-users li', blockMember);
    $(document).on('deleteCustomUser', '#ext48co-settings-users li', deleteCustomUser);

    $(document).on('click', '#ext48co-settings-groups a', switchGroup);
    $(document).on('click', '#ext48co-settings-teams a', swtichTeam);

    $(document).on('keyup', '#ext48co-settings-add', addCustomUser);

    $(document).on('click', '.ext48co-button-settings', function (event) {
      event.preventDefault();
      event.stopPropagation();

      var $dialog = $('#ext48co-settings-dialog');
      if ($dialog.length === 0) {
        buildDialog();
      } else {
        $dialog.trigger('toggleDialog');
      }
    });

    $(document).on('click', '.ext48co-settings-save', function (event) {
      var $dialog = $('#ext48co-settings-dialog');
      $dialog.trigger('saveSettings');
    });

    $(document).on('click', '.ext48co-settings-cancel', function (event) {
      event.preventDefault();
      var $dialog = $('#ext48co-settings-dialog');
      $dialog.trigger('toggleDialog');
    });

    $(document).on('click', '.ext48co-settings-reset', function (event) {
      var $dialog = $('#ext48co-settings-dialog');
      $dialog.trigger('resetSettings');
    });

    $(document).on('click', '#ext48co-settings-menu a', function (event) {
      event.preventDefault();
      var $this = $(this);
      var tab = $this.data('tab');
      $('#ext48co-settings-menu a').removeClass('ext48co-selected');
      $this.addClass('ext48co-selected');
      $('div.ext48co-settings-content[data-tab]').hide();
      $('div.ext48co-settings-content[data-tab="' + tab + '"]').show();
    });

    $(document).on('click', '#ext48co-settings-users li a.ext48co-settings-userblock', function (event) {
      event.preventDefault();
      var $this = $(this);
      var $user = $this.parent('li');
      $user.trigger('blockMember');
    });

    $(document).on('click', '#ext48co-settings-users li a.ext48co-settings-userdelete', function (event) {
      event.preventDefault();
      var $this = $(this);
      var $user = $this.parent('li');
      $user.trigger('deleteCustomUser');
    });

  };

  /**
   * Builds the settings dialog.
   */
  var buildDialog = function () {
    var $dialog = $('<div id="ext48co-settings-dialog" class="ext48co"></div>');

    $dialog.html(
      '<h1>48 Comments Only</h1>' +
      '<div id="ext48co-settings-close"><a href="javascript:void(0)" class="ext48co-settings-cancel"><img src="' + self.ICONS.close + '"></a></div>' +
      '<div id="ext48co-settings-main">' +
      '<div id="ext48co-settings-menu">' +
      '<a href="javascript:void(0)" data-tab="userlist">' + self._('Members & Custom Users') + '</a>' +
      '<a href="javascript:void(0)" data-tab="extraction">' + self._('Extraction') + '</a>' +
      '<a href="javascript:void(0)" data-tab="display">' + self._('Display') + '</a>' +
      '</div>' +

      '<div class="ext48co-settings-content" data-tab="extraction">' +
      /* 2015.12.07 @fronoske 自動ロードは機能しないので選択肢から削除
      '<p><label>' + self._('Extraction:') + '</label> <select data-key="commentLoadTrigger"><option value="mouseenter">' + self._('Cursor moving') + '</option><option value="DOMNodeInserted">' + self._('Auto load') + '</option></select></p>' +
      */
      '<p><label>' + self._('Extraction:') + '</label> <select data-key="commentLoadTrigger"><option value="mouseenter" selected>' + self._('Cursor moving') + '</option></select></p>' +
      '<p><label>' + self._('Interval:') + '</label> <input type="text" size="5" data-key="commentLoadInterval"> ms</p>' +
      '<p><label>' + self._('Delay:') + '</label> <input type="text" size="5" data-key="triggerDelay"> ms</p>' +
      '<p><input type="checkbox" data-key="highlightMyComments"> <label>' + self._('Load my comments when I logged in.') + '</label></p>' +
      '<p><input type="checkbox" data-key="loadSharers"> <label>' + self._('Load post share information.') + '</label></p>' +
      '</div>' +

      '<div class="ext48co-settings-content" data-tab="display">' +
      '<p><label>' + self._('Comment Style:') + '</label> <select data-key="commentStyle"><option value="ballon">' + self._('Speech balloon style') + '</option><option value="normal">' + self._('Normal style') + '</option></select></p>' +
      '<p><label>' + self._('Comment Position:') + '</label> <select data-key="commentsPosition"><option value="content">' + self._('Bottom of post content') + '</option><option value="list">' + self._('Top of all comments') + '</option><option value="popup">' + self._('Popup') + '</option></select></p>' +
      '<p><label>' + self._('Colors:') + '</label> <select data-key="colors">' +
      '<option value="#FFF2F6,#FFE6EE,#FF3377" style="background: #FFE6EE; color: #FF3377;">' + self._('Pink') + '</option>' +
      '<option value="#FFF2F4,#FF485E,#FFFFFF" style="background: #FF485E; color: #FFFFFF;">' + self._('Red') + '</option>' +
      '<option value="#F2FFF7,#38D171,#FFFFFF" style="background: #38D171; color: #FFFFFF;">' + self._('Green') + '</option>' +
      '<option value="#F2F7FF,#6CA2FF,#FFFFFF" style="background: #6CA2FF; color: #FFFFFF;">' + self._('Blue') + '</option>' +
      '<option value="#FFFDF2,#FFEB4A,#666666" style="background: #FFEB4A; color: #666666;">' + self._('Yellow') + '</option>' +
      '<option value="#F2F2F2,#9D9D9D,#333333" style="background: #9D9D9D; color: #333333;">' + self._('Gray') + '</option>' +
      '</select></p>' +
      '<p><input type="checkbox" data-key="autoExpand"> <label>' + self._('Expand all comments by default.') + '</label></p>' +
      '<p><input type="checkbox" data-key="hideStreamComments"> <label>' + self._('Hide other comments in stream page.') + '</label></p>' +
      '<p><input type="checkbox" data-key="displayCommentNumber"> <label>' + self._('Display comment number.') + '</label></p>' +
      '<p><input type="checkbox" data-key="displayCommentHumanTime"> <label>' + self._('Display comment time difference.') + '</label></p>' +
      '</div>' +

      '<div class="ext48co-settings-content" data-tab="userlist">' +
      '<p>' + self._('Comments posted by users in list will be displayed.') + '</p>' +
      '<p class="ext48co-desc">' +
      // 2015.02.14 @fronoske メンバーデータの取得日時の表示を改善
      '<a href="' + self.SPREADSHEET_URL + '/">' + self._('Latest update of member data:') + ' ' + '<span id="ext48co-settings-dataupdated"></span></a>' +
      '<br/>' +
      '<span>' + self._('Latest update on server:') + ' ' + '<span id="ext48co-settings-db-updated"></span></span>' +
      /* end */
      '</p>' +
      '<div id="ext48co-settings-groups"></div>' +
      '<div id="ext48co-settings-teams"></div>' +
      '<ul id="ext48co-settings-users"></ul>' +
      '</div>' +
      '</div>' +

      '<div id="ext48co-settings-info">' +
      '<strong>48 Comments Only</strong>' +
      '<br>' +
      'Written by Ming-Hsien Lin (akiratw), 2011-2015. ' +
      'Revised by <a href="./112869918300681260602" oid="112869918300681260602">@fronoske</a>, 2015.<br/>' +
      'Iconset <a href="http://www.entypo.com/">Entypo</a>.' +
      '</div>' +

      '<div id="ext48co-settings-buttons">' +
      '<button class="ext48co ext48co-settings-reset">' + self._('Reset') + '</button>' +
      '<button class="ext48co ext48co-settings-cancel">' + self._('Cancel') + '</button>' +
      '<button class="ext48co ext48co-main ext48co-settings-save">' + self._('Save') + '</button>' +
      '</div>');

    $('body').append($dialog);
    $('#ext48co-update').trigger('checkUpdate');
    $('#ext48co-settings-menu a').eq(0).click();
    $dialog.trigger('readSettings')
      .trigger('toggleDialog');
  };

  /**
   * Reads all settings and user data to dialog.
   */
  var readSettings = function (event) {
    var $dialog = $(this);
    var $fields = $dialog.find('[data-key]');

    $fields.each(function () {
      var $field = $(this);
      var value = self._settings[$field.data('key')];

      if ($field.prop('tagName') == 'INPUT') {
        if ($field.prop('type') == 'text') $field.val(value);
        if ($field.prop('type') == 'checkbox') $field.prop('checked', value);
      } else if ($field.prop('tagName') == 'SELECT') {
        if (Array.isArray(value)) value = value.join(',');
        $field.find('option[value="' + value + '"]').prop('selected', 'selected');
      }
    });

    var $groups = $('#ext48co-settings-groups');
    var $teams = $('#ext48co-settings-teams');
    var $users = $('#ext48co-settings-users');
    var $updated = $('#ext48co-settings-dataupdated');
    var $dbUpdated = $('#ext48co-settings-db-updated');
    var ids = [];

    $updated.text(
    self._members.updated.toHumanTimeDiff(self._('humantime')) + ' ' +
      '(' + self._members.updated.toShortString() + ')');
    $dbUpdated.text(
    self._members.dbUpdated.toHumanTimeDiff(self._('humantime')) + ' ' +
      '(' + self._members.dbUpdated.toShortString() + ')');
    $groups.empty()
      .append(
      '<a href="javascript:void(0)" data-group="all">' + self._('All') + '</a>' +
      '<a href="javascript:void(0)" data-group="custom">' + self._('Custom') + '</a>');
    $teams.empty();
    $users.empty()
      .append(
      '<li data-group="custom">' +
      '<img src="' + self.ICONS.addUser + '"> ' +
      '<input type="text" id="ext48co-settings-add" class="ext48co" size="50" placeholder="' + self._('Add a user to list by ID or Profile URL.') + '">' +
      '</li>');

    for (var group in self._members.groups) {
      $groups.append('<a href="javascript:void(0)" data-group="' + escape(group) + '">' + self._(group) + '</a>');

      var teams = self._members.groups[group];
      teams.forEach(function (element, index) {
        $teams.append('<a href="javascript:void(0)" data-group="' + escape(group) + '" data-team="' + escape(element) + '">' + self._(element) + '</a>');
      });
    }

    for (var id in self._members.members) {
      ids.push(id);

      var member = self._members.members[id];
      var $member = $('<li>').appendTo($users);
      $member.trigger('addUser', [id, member.name, '', '', member.group, member.team, false]);

      if (self._settings.blockIDs.indexOf(id) != -1) $member.trigger('blockMember');
    }

    self._settings.gplusIDs.forEach(function (element, index) {
      if (ids.indexOf(element) != -1) return;
      ids.push(element);

      var $user = $('<li>').insertAfter($users.find('li').eq(0));
      $user.trigger('addUser', [element, element, '', '', 'custom', 'custom', true]);
    });

    $groups.find('a').eq(0).click();

    self.API.lookupUsers(ids, function (response) {
      if (!response.success) return;
      var data = response.data;
      for (var id in data) {
        var $user = $users.find('li[data-id="' + id + '"]');
        $user.trigger('editUser', [id, data[id].name, data[id].photo, data[id].description]);
      }
    });
  };

  /**
   * Saves all settings.
   */
  var saveSettings = function (event) {
    var $dialog = $(this);
    var $fields = $dialog.find('[data-key]');

    $fields.each(function () {
      var $field = $(this);
      var key = $field.data('key');

      if ($field.prop('tagName') == 'INPUT') {
        if ($field.prop('type') == 'text') self._settings[key] = $field.val();
        if ($field.prop('type') == 'checkbox') self._settings[key] = $field.prop('checked');
      } else if ($field.prop('tagName') == 'SELECT') {
        var value = $field.find('option:selected').prop('value');
        values = value.split(',');
        if (values.length == 1) {
          self._settings[key] = values[0];
        } else {
          self._settings[key] = values;
        }
      }
    });

    self._settings.gplusIDs = [];
    self._settings.blockIDs = [];

    var $users = $dialog.find('#ext48co-settings-users li');
    $users.each(function () {
      var $user = $(this);
      if ($user.attr('data-group') == 'custom' && $user.attr('data-id')) {
        self._settings.gplusIDs.push($user.attr('data-id'));
      } else if ($user.attr('data-block')) {
        self._settings.blockIDs.push($user.attr('data-id'));
      }
    });

    self._writeSettings();
    location.reload(true);
  };

  /**
   * Resets all settings.
   */
  var resetSettings = function (event) {
    var answer = confirm(self._('Do you want to reset all options?'));
    if (!answer) return;

    localStorage.removeItem(self.SETTINGS_KEY);
    location.reload(true);
  };

  /**
   * Adds a user to list.
   */
  var addUser = function (event, id, name, photo, description, group, team, deletable) {
    var $user = $(this);
    var $list = $('#ext48co-settings-users');

    if ($list.find('li[data-id="' + id + '"]').length !== 0) return;

    $user.attr('data-id', id)
      .attr('data-group', escape(group))
      .attr('data-team', escape(team))
      .html(
    (deletable ?
      '<a href="javascript:void(0)" class="ext48co-settings-userbutton ext48co-settings-userdelete" title="' + self._('Delete this user from list.') + '"><img src="' + self.ICONS.close + '"></a>' :
      '<a href="javascript:void(0)" class="ext48co-settings-userbutton ext48co-settings-userblock" title="' + self._('Block this user from comment displayed.') + '"><img src="' + self.ICONS.block + '"></a>') +
      '<img class="ext48co-settings-photo" src="' + photo + '">' +
      '<a class="ext48co-settings-name" href="./' + id + '" oid="' + id + '">' + name + '</a>' +
      '<span>' + self._('Loading...') + '</span>');
  };

  /**
   * Edits user information in list.
   */
  var editUser = function (event, id, name, photo, description) {
    var $user = $(this);

    $user.find('img.ext48co-settings-photo').prop('src', photo + '?sz=24');
    $user.find('a.ext48co-settings-name').eq(0).text(name);
    $user.find('span').eq(0).text(description || self._(unescape($user.attr('data-team'))));
  };

  /**
   * Blocks a member from comment extraction.
   */
  var blockMember = function (event) {
    var $user = $(this);
    if ($user.attr('data-block')) {
      $user.removeAttr('data-block')
        .css({
        textDecoration: ''
      })
        .find('a.ext48co-settings-userblock img').prop('src', self.ICONS.block);
    } else {
      $user.attr('data-block', true)
        .css({
        textDecoration: 'line-through'
      })
        .find('a.ext48co-settings-userblock img').prop('src', self.ICONS.blocked);
    }
  };

  /**
   * Switches a group to display.
   */
  var switchGroup = function (event) {
    event.preventDefault();

    var $tab = $(this);
    var $groups = $('#ext48co-settings-groups');
    var $teams = $('#ext48co-settings-teams');
    var $list = $('#ext48co-settings-users');
    var group = $tab.attr('data-group');

    $teams.find('a').hide();
    $list.find('li').hide();

    if (group == 'all') {
      $list.find('li').show();
    } else {
      $list.find('li[data-group="' + group + '"]').show();
      $teams.find('a[data-group="' + group + '"]').show();
    }

    $groups.find('a').removeClass('ext48co-selected');
    $teams.find('a').removeClass('ext48co-selected');
    $tab.addClass('ext48co-selected');

    $list.animate({
      scrollTop: 0
    }, 200);
  };

  /**
   * Switches a team to display.
   */
  var swtichTeam = function (event) {
    event.preventDefault();

    var $tab = $(this);
    var $groups = $('#ext48co-settings-groups');
    var $teams = $('#ext48co-settings-teams');
    var $list = $('#ext48co-settings-users');
    var group = $tab.attr('data-group');
    var team = $tab.attr('data-team');

    $teams.find('a').hide();
    $list.find('li').hide();

    $teams.find('a[data-group="' + group + '"]').show();
    $list.find('li[data-group="' + group + '"][data-team="' + team + '"]').show();

    $teams.find('a').removeClass('ext48co-selected');
    $tab.addClass('ext48co-selected');

    $list.animate({
      scrollTop: 0
    }, 200);
  };

  /**
   * Adds a custom user by ID or Profile URL.
   */
  var addCustomUser = function (event) {
    var input = $(this).val();
    var $list = $('#ext48co-settings-users');

    if (!input.match(/(\d{21})/)) return;

    var id = input.match(/(\d{21})/)[1];
    if ($list.find('li[data-id="' + id + '"]').length !== 0) return;

    var $user = $('<li>').insertAfter($list.find('li').eq(0));
    $user.css('background', '#FFC')
      .trigger('addUser', [id, id, '', '', 'custom', 'Custom', true]);

    $(this).val('');

    self.API.lookupUsers(id, function (response) {
      if (response.success) {
        var data = response.data[id];
        $user.trigger('editUser', [id, data.name, data.photo, data.description]);
      } else {
        $user.fadeOut(200, function () {
          $(this).remove();
        });
      }
    });
  };

  /**
   * Deletes a custom user.
   */
  var deleteCustomUser = function (event) {
    var $user = $(this);
    $user.fadeOut(200, function () {
      $(this).remove();
    });
  };

  /**
   * Displays or hides the settings dialog.
   */
  var toggleDialog = function (event) {
    var $dialog = $(this);
    $dialog.css('left', ($(window).width() - $dialog.outerWidth()) / 2)
      .slideToggle(200);
  };

  /**
   * Checks userscript update and display version information.
   */
  var checkUpdate = function (event) {
    var $this = $(this);
    self._checkUpdate(function (response) {
      if (response.success) {
        if (response.data.isUpdateAvailable) {
          $this.html(' - ' + '<a href="' + response.data.scriptURL + '">' + self._('A newer version (%) is found. Click here to install.', response.data.version) + '</a>');
        } else {
          $this.html(' - ' + self._('You are using the latest version.'));
        }
      }
    });
  };

  addStyle();
  bindEvents();
  initButton();
};

CommentsOnly.prototype._initData = function (callback) {
  var self = this;
  var _callback = callback;

  /**
   * Fetches current login user data.
   */
  var getLoginUserData = function () {
    self.API.lookupCurrentUser(function (response) {
      if (response.error) return;

      self._login_user.id = response.data.id;
      self._login_user.name = response.data.name;
      self._login_user.photo = response.data.photo;
    });
  };

  /**
   * Builds Regex for comments extraction and purify user settings.
   */
  var buildRegex = function (ids) {
    self._settings.gplusIDs.forEach(function(element, index) {
      if (ids.indexOf(element) == -1) {
        ids.push(element);
      }
    });

    self._idRegex = new RegExp('(' + ids.join('|') + ')');
  };

  /**
   * Retrieves member data.
   */
  var retrievesMemberData = function (data) {
    data.people.forEach(function (element, index) {
      var id = element.id;
      var name = element.name;
      var group = element.group;
      var team = element.team;

      self._members.members[id] = {
        id: id,
        name: name,
        group: group,
        team: team
      };

      if (!Array.isArray(self._members.groups[group])) {
        self._members.groups[group] = [];
      }
      if (self._members.groups[group].indexOf(team) == -1) {
        self._members.groups[group].push(team);
      }
    });

    self._members.updated = new Date(data.updated);
    self._members.dbUpdated = new Date(data.dbUpdated);
    self._settings.memberData = data;
    self._writeSettings();

    buildRegex(data.ids);
  };

  var getJSON = function () {
    $.ajax({
      // url: 'https://script.google.com/macros/s/AKfycbzg_KpbULRmyY-_TOjU0UPrGyypP8bl4PWg_ZcdXEEvtHmAtck/exec',
      // url: 'https://script.google.com/macros/s/AKfycbxi8qgVpXm39lX9HH4Xv5FUyS86Fuqcs2nifrCCFGs3nRIPqjc/exec',
      url: 'https://spreadsheets.google.com/feeds/list/1uWMxVR0MjwL2p-N2k0WuG096qIDtSA92tsTSjKV3w_8/od6/public/values?alt=json',
      type: 'GET',
      dataType: 'json',
      success: function (data) {
        // load JSON directly (not via script) from spreadsheet
        var data2 = {
          "kind": "memberlist",
          "dbUpdated": data['feed']['updated']['$t'],
          "updated": new Date(),
          "etag": "d41d8cd98f00b204e9800" + (new Date(data['feed']['updated']['$t'])).getTime().toString(16),
          "totalItems": data['feed']['entry'].length,
          "people": [],
          "ids": []
        };
        var ids = [];
        for(var i=0; i < data['feed']['entry'].length; i++){
          var _entry = data['feed']['entry'][i];
          var _id = _entry['gsx$id']['$t'];
          var _name = _entry['gsx$name']['$t'];
          var _group = _entry['gsx$group']['$t'];
          var _team = _entry['gsx$team']['$t'];
          var _status = _entry['gsx$status']['$t'];
          if (_id != "" && _status == ""){ // statusが "removed" などのIDは登録しない
            ids.push(_id);
            data2['people'].push( {"id": _id, "name": _name, "group": _group, "team": _team, "status": _status } );
          }
        }
        data2["ids"] = ids.filter(function (x, i, self) { return self.indexOf(x) === i; });
        retrievesMemberData(data2);
        getLoginUserData();
        _callback();
      }
    });
  };

  if (self._settings.memberData.updated && self._settings.memberData.dbUpdated) {
    var updated = new Date(self._settings.memberData.updated),
        now = new Date();
    
    if (now.getTime() - updated.getTime() < 86400000 * 7) {
      retrievesMemberData(self._settings.memberData);
      getLoginUserData();
      _callback();
    } else {
      getJSON();
    }
  } else {
    getJSON();
  }

};

CommentsOnly.prototype._localize = function () {
  var langs = {
    'en': {
      'name': 'English',
      'humantime': ['Just now', '1 minute ago', '% minutes ago', '1 hour ago', '% hours ago', '1 day ago', '% days ago'],
      '48 Comments Only Settings': '48 Comments Only Settings',
      'Collapse or expand comments': 'Collapse or expand comments',
      'Loading...': 'Loading...',
      'HTTP % - %': 'HTTP % - %',
      'Error occurred.': 'Error occurred.',
      ' (edited)': ' (edited)',
      'Edit': 'Edit',
      'Copy': 'Copy',
      ', ': ', ',
      '% comments': '% comments',
      'No comment from members': 'No comment from members',
      '% shared this post.': '% shared this post.',
      'Save': 'Save',
      'Delete': 'Delete',
      'Cancel': 'Cancel',
      'Do you want to permanently delete this comment?': 'Do you want to permanently delete this comment?',
      'Members & Custom Users': 'Members & Custom Users',
      'Extraction': 'Extraction',
      'Display': 'Display',
      'Extraction:': 'Extraction:',
      'Cursor moving': 'Cursor moving',
      'Auto load': 'Auto load',
      'Interval:': 'Interval:',
      'Delay:': 'Delay:',
      'Load my comments when I logged in.': 'Load my comments when I logged in.',
      'Load post share information.': 'Load post share information.',
      'Comment Style:': 'Comment Style:',
      'Speech balloon style': 'Speech balloon style',
      'Normal style': 'Normal style',
      'Comment Position:': 'Comment Position:',
      'Bottom of post content': 'Bottom of post content',
      'Top of all comments': 'Top of all comments',
      'Popup': 'Popup',
      'Colors:': 'Colors:',
      'Pink': 'Pink',
      'Red': 'Red',
      'Green': 'Green',
      'Blue': 'Blue',
      'Yellow': 'Yellow',
      'Gray': 'Gray',
      'Expand all comments by default.': 'Expand all comments by default.',
      'Hide other comments in stream page.': 'Hide other comments in stream page.',
      'Display comment number.': 'Display comment number.',
      'Display comment time difference.': 'Display comment time difference.',
      'Comments posted by users in list will be displayed.': 'Comments posted by users in list will be displayed.',
      'Latest update of member data:': 'Latest update of member data:',
      'Latest update on server:': 'Latest update on server:',
      'Reset': 'Reset',
      'All': 'All',
      'Custom': 'My List',
      'Official': 'Official',
      'Page': 'Page',
      'Staff': 'Staff',
      'Trainee': 'Trainee',
      'Other': 'Other',
      'OG': 'Former members',
      'Add a user to list by ID or Profile URL.': 'Add a user to list by ID or Profile URL.',
      'Do you want to reset all options?': 'Do you want to reset all options?',
      'Delete this user from list.': 'Delete this user from list.',
      'Block this user from comment displayed.': 'Block this user from comment displayed.',
      'A newer version (%) is found. Click here to install.': 'A newer version (%) is found. Click here to install.',
      'You are using the latest version.': 'You are using the latest version.',
      // 2015.12.14 @fronoske
      "% people +1'd this comment": "% people +1'd this comment"
    },
    'ja': {
      'name': '日本語',
      '48 Comments Only Settings': '48 Comments Only 設定',
      'humantime': ['たった今', '1 分前', '% 分前', '1 時間前', '% 時間前', '1 日前', '% 日前'],
      'Collapse or expand comments': 'コメントを折りたたむ/展開する',
      'Loading...': '読み込んでいます…',
      'HTTP % - %': 'HTTP % - %',
      'Error occurred.': 'エラーが発生しました。',
      ' (edited)': ' (編集)',
      'Edit': '編集',
      'Copy': 'コピー',
      ', ': '、',
      '% comments': '% 件のコメント',
      'No comment from members': 'メンバーのコメントが見つかりません。',
      '% shared this post.': '% がこの投稿を共有しました。',
      'Save': '変更保存',
      'Delete': '削除',
      'Cancel': 'キャンセル',
      'Do you want to permanently delete this comment?': 'このコメントを完全に削除してもよろしいですか?',
      'Members & Custom Users': 'メンバーとマイリスト',
      'Extraction': 'コメント抽出設定',
      'Display': '表示設定',
      'Extraction:': '抽出:',
      'Cursor moving': 'カーソルを合わせた時',
      'Auto load': '自動ロード',
      'Interval:': '抽出間隔:',
      'Delay:': '抽出遅延:',
      'Load my comments when I logged in.': 'ログインした場合は自分のコメントを表示する',
      'Load post share information.': '共有情報を表示する',
      'Comment Style:': 'コメント仕様:',
      'Speech balloon style': 'ふきだし',
      'Normal style': 'ノーマル',
      'Comment Position:': 'コメントリストの位置:',
      'Bottom of post content': '投稿内容の下',
      'Top of all comments': 'すべてのコメントの上',
      'Popup': '飛び出し',
      'Colors:': '色選択:',
      'Pink': 'ピンク',
      'Red': '赤',
      'Green': '緑',
      'Blue': '青',
      'Yellow': '黄',
      'Gray': 'グレー',
      'Expand all comments by default.': 'デフォルトでコメントを展開する',
      'Hide other comments in stream page.': 'ストリームでメンバー以外のコメントを非表示にする',
      'Display comment number.': 'コメントの番号を表示する',
      'Display comment time difference.': 'コメントの時間差を表示する',
      'Comments posted by users in list will be displayed.': 'リストにいるメンバーとユーザーの投稿したコメントが表示されます。',
      'Latest update of member data:': 'メンバーデータの更新日時:',
      'Latest update on server:': 'サーバーデータの更新日時:',
      'Reset': 'リセット',
      'All': 'すべて',
      'Custom': 'マイリスト',
      'Official': '運営/関係者',
      'Page': 'ページ',
      'Staff': 'スタッフ',
      'Trainee': '研究生',
      'Other': 'その他',
      'OG': '元メンバー',
      'Add a user to list by ID or Profile URL.': 'ID またはプロフィールの URL でユーザーを追加する…',
      'Do you want to reset all options?': 'すべての設定をリセットしますか?',
      'Delete this user from list.': 'リストからこのユーザーを削除する。',
      'Block this user from comment displayed.': 'このユーザーのコメントを非表示にする。',
      'A newer version (%) is found. Click here to install.': '新しいバージョン(%)があります。',
      'You are using the latest version.': '最新バージョン',
      // 2015.12.14 @fronoske
      "% people +1'd this comment": '% 人がこのコメントを +1 しました'
    },
    'zh-TW': {
      'name': '正體中文',
      '48 Comments Only Settings': '48 Comments Only 設定',
      'humantime': ['剛剛', '1 分鐘前', '% 分鐘前', '1小時前', '% 小時前', '1 天前', '% 天前'],
      'Collapse or expand comments': '收合/展開所有留言',
      'Loading...': '正在讀取中…',
      'HTTP % - %': 'HTTP % - %',
      'Error occurred.': '發生錯誤。',
      ' (edited)': ' (編輯)',
      'Edit': '編輯',
      'Copy': '複製',
      ', ': '、',
      '% comments': '% 則留言',
      'No comment from members': '找不到來自成員的留言。',
      '% shared this post.': '% 分享了這則訊息。',
      'Save': '儲存變更',
      'Delete': '刪除',
      'Cancel': '取消',
      'Do you want to permanently delete this comment?': '你確定要永久刪除這則留言嗎?',
      'Members & Custom Users': '成員與自訂使用者',
      'Extraction': '留言載入設定',
      'Display': '顯示設定',
      'Extraction:': '載入:',
      'Cursor moving': '游標移過時',
      'Auto load': '自動載入',
      'Interval:': '載入間隔:',
      'Delay:': '載入延遲:',
      'Load my comments when I logged in.': '登入時顯示我發表的留言',
      'Load post share information.': '顯示成員間的分享資訊',
      'Comment Style:': '留言顯示樣式:',
      'Speech balloon style': '對話框',
      'Normal style': '一般',
      'Comment Position:': '留言列表位置:',
      'Bottom of post content': '訊息內文的下方',
      'Top of all comments': '所有留言的最上方',
      'Popup': '彈出式',
      'Colors:': '顏色設定:',
      'Pink': '粉紅',
      'Red': '紅',
      'Green': '綠',
      'Blue': '藍',
      'Yellow': '黃',
      'Gray': '灰',
      'Expand all comments by default.': '預設展開所有留言',
      'Hide other comments in stream page.': '在訊息串中隱藏其他非成員的留言',
      'Display comment number.': '顯示留言編號',
      'Display comment time difference.': '顯示留言時間差',
      'Comments posted by users in list will be displayed.': '在以下清單內的成員和使用者所發表的留言會顯示出來。',
      'Latest update of member data:': '成員資料最後更新於:',
      'Latest update on server:': 'Lateset update on server',
      'Reset': '重設',
      'All': '全部',
      'Custom': '自訂',
      'Official': '官方與關係者',
      'Page': '專頁',
      'Staff': '工作人員',
      'Trainee': '研究生',
      'Other': '其他',
      'OG': '已畢業成員',
      'Add a user to list by ID or Profile URL.': '以 ID 或個人檔案網址來新增使用者…',
      'Do you want to reset all options?': '你確定要重設所有設定嗎?',
      'Delete this user from list.': '刪除這個使用者。',
      'Block this user from comment displayed.': '不要顯示此成員的留言。',
      'A newer version (%) is found. Click here to install.': '有新的版本(%)。',
      'You are using the latest version.': '最新版本',
       // 2015.12.14 @fronoske
      "% people +1'd this comment": '% 人 +1 了這則留言'
    },
    'zh-CN': {
      'name': '简体中文',
      '48 Comments Only Settings': '48 Comments Only 设定',
      'humantime': ['刚刚', '1 分钟前', '% 分钟前', '1小时前', '% 小时前', '1 天前', '% 天前'],
      'Collapse or expand comments': '收合/展开所有评论',
      'Loading...': '正在读取中…',
      'HTTP % - %': 'HTTP % - %',
      'Error occurred.': '发生错误。',
      ' (edited)': ' (编辑)',
      'Edit': '编辑',
      'Copy': '复制',
      ', ': '、',
      '% comments': '% 条评论',
      'No comment from members': '找不到来自成员的评论。',
      '% shared this post.': '% 分享了这则信息。',
      'Save': '保存更改',
      'Delete': '删除',
      'Cancel': '取消',
      'Do you want to permanently delete this comment?': '要永久性删除此评论吗?',
      'Members & Custom Users': '成员与自订用户',
      'Extraction': '评论加载设定',
      'Display': '显示设定',
      'Extraction:': '加载:',
      'Cursor moving': '鼠标移过时',
      'Auto load': '自动加载',
      'Interval:': '加载间隔:',
      'Delay:': '加载延迟:',
      'Load my comments when I logged in.': '登入时显示我发表的评论',
      'Load post share information.': '显示成员间的分享信息',
      'Comment Style:': '评论显示样式:',
      'Speech balloon style': '对话框',
      'Normal style': '一般',
      'Comment Position:': '评论列表位置:',
      'Bottom of post content': '信息内文的下方',
      'Top of all comments': '所有评论的最上方',
      'Colors:': '颜色设定:',
      'Pink': '粉红',
      'Red': '红',
      'Green': '绿',
      'Blue': '蓝',
      'Yellow': '黄',
      'Gray': '灰',
      'Expand all comments by default.': '默认展开所有评论',
      'Hide other comments in stream page.': '在讯息串中隐藏其他非成员的评论',
      'Display comment number.': '显示评论编号',
      'Display comment time difference.': '显示评论时间差',
      'Comments posted by users in list will be displayed.': '在以下清单内的成员和用户所发表的评论会显示出来。',
      'Latest update of member data:': '成员数据最后更新于:',
      'Latest update on server:': 'Lateset update on server',
      'Reset': '重置',
      'All': '全部',
      'Custom': '自订',
      'Official': '官方与关系者',
      'Page': '专页',
      'Staff': '工作人员',
      'Trainee': '研究生',
      'Other': '其他',
      'OG': '已毕业成员',
      'Add a user to list by ID or Profile URL.': '以 ID 或个人档案网址来新增用户…',
      'Do you want to reset all options?': '你确定要重置所有设定吗?',
      'Delete this user from list.': '删除这个用户。',
      'Block this user from comment displayed.': '不要显示此成员的评论。',
      'A newer version (%) is found. Click here to install.': '有新的版本(%)。',
      'You are using the latest version.': '最新版本',
       // 2015.12.14 @fronoske
      "% people +1'd this comment": '有 % 个人为此评论 +1 了'
    }
  };

  var langCode = $('html').prop('lang');
  this._lang = langs[langCode] || langs[navigator.language] || langs.ja;
};

CommentsOnly.prototype._ = function (string, replace) {
  var output = this._lang[string] || string;

  if (replace) {
    if (!Array.isArray(replace)) replace = [replace];

    replace.forEach(function (element, index) {
      output = output.replace('%', element);
    });
  }

  return output;
};

CommentsOnly.prototype._readSettings = function () {
  var settings = localStorage.getItem(this.SETTINGS_KEY);
  if (settings) {
    this._settings = $.extend(this._settings, JSON.parse(settings));
  }
};

CommentsOnly.prototype._writeSettings = function () {
  var settings = JSON.stringify(this._settings);
  localStorage.setItem(this.SETTINGS_KEY, settings);
};

CommentsOnly.prototype._addStyle = function (id, css) {
  GM_addStyle(css);
};


var GooglePlusAPI = function () {
  // ---------- CONSTANTS ----------
  // API URLs.
  this.ACTIVITY_URL = 'https://plus.google.com/_/stream/getactivity/';
  this.SHARERS_URL = 'https://plus.google.com/_/stream/getsharers/';
  this.LOOKUP_URL = 'https://plus.google.com/_/socialgraph/lookup/hovercards/';
  this.INITIAL_DATA_URL = 'https://plus.google.com/_/initialdata?key=14';
  this.COMMENT_EDIT_URL = 'https://plus.google.com/_/stream/editcomment/';
  this.COMMENT_DELETE_URL = 'https://plus.google.com/_/stream/deletecomment/';
  this.COMMENT_PLUSONE_URL = 'https://plus.google.com/_/plusone';

  // 2015.12.10 @fronosoke
  this.OLD_LAYOUT = !$("body").attr("jscontroller"); // 2015年秋のレイアウト変更前かどうか
  this.GET_SESSION_URL = 'https://plus.google.com/u/0/_/notifications/frame'
  this.PLUSONE_USERS_URL = 'https://plus.google.com/_/common/getpeople/'
  // ref. https://github.com/mohamedmansour/google-plus-extension-jsapi
  // ---------- Private Fields ----------
  this._session = null;
};

/**
 * Parses Google+ JSON string.
 *
 * @param {string} input A JSON string returned by Google+.
 * @return {Object} A parsed JSON object.
 */
GooglePlusAPI.prototype._parseJSON = function (input) {
  // var jsonString = input.replace(/\[,/g, '[null,');
  // jsonString = jsonString.replace(/,\]/g, ',null]');
  // jsonString = jsonString.replace(/,,/g, ',null,');
  // jsonString = jsonString.replace(/,,/g, ',null,');
  // jsonString = jsonString.replace(/\{(\d+):/, '{"$1":');
  //
  // return JSON.parse(jsonString);

  try {
    return eval('(' + input +')');
  } catch (error) {
    return {};
  }
};

/**
 * Fires callback safely.
 *
 * @param {boolean} isSuccess Whether the request is successful.
 * @param {Object} data Data to send in the callback.
 * @param {function (Object)} callback A callback to fire back.
 */
GooglePlusAPI.prototype._fireCallback = function (isSuccess, data, callback) {
  if (!callback) return;
  callback({
    success: isSuccess,
    data: data
  });
};

/**
 * Sends an HTTP request to Google + server.
 *
 * @param {string} url An URL to request.
 * @param {?string} postData A string of data to be sent to server.
 * @param {function (Object)} callback A callback.
 */
GooglePlusAPI.prototype._sendRequest = function (url, postData, callback) {
  var self = this;

  var doSuccess = function (data, textStatus, jqXHR) {
    if (!data && jqXHR.status === 200) {
      var responseText = jqXHR.responseText.substring(4);
      var results = self._parseJSON(responseText);
      callback(Array.isArray(results) ? results[0] : results);
    } else if (data && jqXHR.status === 200) {
      callback(data);
    }
  };

  var doError = function (jqXHR, textStatus, errorThrown) {
    if (textStatus === 'parsererror') {
      if (jqXHR.status === 200) {
        doSuccess(null, textStatus, jqXHR);
      }
      return;
    }
    callback({
      error: errorThrown || jqXHR.status || 'error',
      text: textStatus,
      responseText: jqXHR.responseText
    });
  };

  return $.ajax({
    url: url,
    type: postData ? 'POST' : 'GET',
    data: postData || null,
    dataType: 'json',
    success: doSuccess,
    error: doError
  });
};

/**
 * Verifies an HTTP request is successful. If not, fire callback.
 *
 * @param {Object} response An response object returned by _sendRequest().
 * @param {function (Object)} callback A callback.
 * @return {boolean} Whether the request is successful.
 */
GooglePlusAPI.prototype._isRequestSuccess = function (response, callback) {
  if (response.error) {
    this._fireCallback(false, response, callback);
    return false;
  } else {
    return true;
  }
};

/**
 * Fetches Google+ current login user session.
 *
 * @param {boolean} reset Force re-fetch.
 * @return {string} Google+ login user session.
 */
GooglePlusAPI.prototype._getSession = function (param) {
  var self = this;
  param = $.extend({reset: false, async: true}, param);
  if (this._session && !param.reset) return this._session;

  var html = $('body').html();
  var searchFor = ',"https://csi.gstatic.com/csi","';
  var startIndex = html.indexOf(searchFor);
  
  if (startIndex !== -1) {
    var remaining = html.substring(startIndex + searchFor.length);
    this._session = remaining.substring(0, remaining.indexOf('"'));
    console.log("[48CO] (old layout) session=" + this._session);
    return this._session;
  }
  // 2015.12.13 @fronoske 新レイアウト対応
  var ajax_param = {
        url: this.GET_SESSION_URL,
        type:"get",
        async: param.async,
        dataType:"html"
      };
  if (!this._session){
    if (param.async == false){ // ブロッキング
      // console.log("[48CO] Sorry. blocking AJAX start");
      var text = $.ajax(ajax_param).responseText;
      var searchFor = ',"https://csi.gstatic.com/csi","';
      var startIndex = text.indexOf(searchFor);
      if (startIndex !== -1) {
        var remaining = text.substring(startIndex + searchFor.length);
        this._session = remaining.substring(0, remaining.indexOf('"'));
        // console.log("[48CO] getSession sync: " + this._session);
      }
    } else { // ノンブロッキング(取得済みなら上でreturnするのでまずここは通らない)
      $.ajax(ajax_param).then( function(response){
        // console.log("[48CO] getSession async success");
        var searchFor = ',"https://csi.gstatic.com/csi","';
        var startIndex = response.indexOf(searchFor);
        if (startIndex !== -1) {
          var remaining = response.substring(startIndex + searchFor.length);
          this._session = remaining.substring(0, remaining.indexOf('"'));
          // console.log("[48CO] getSession async: " + this._session);
        }
      });
    }
  }
  // console.log("[48CO] session=" + this._session);
  return this._session;
};

/**
 * Parses a Google+ post array to object.
 *
 * @param {Array} postArray A Google+ post array.
 * @return {Object} A parsed post data object.
 */
GooglePlusAPI.prototype._parsePost = function (postArray) {
  var self = this;
  var post = {
    id: postArray[8],
    comments: [],
    sharedFrom: postArray[39]
  };

  postArray[7].forEach(function (element, index) {
    post.comments.push(self._parseComment(element));
  });

  return post;
};

/**
 * Parses a Google+ comment array to object.
 *
 * @param {Array} commentArray A Google+ comment array.
 * @return {Object} A parsed comment data object.
 */
GooglePlusAPI.prototype._parseComment = function (commentArray) {
  // 2015.02.14 @fronoske G+の仕様変更に伴う修正
  /*
  console.log(commentArray);
  if(commentArray[6] == "108806390753406890782")
    for(var i=0; i < commentArray.length; i++){
      console.error(i);
      console.error(commentArray[i]);
    }*/
  var contentStr = commentArray[2] || '';
  var contentArray = commentArray[27] && commentArray[27][0] || [];
  /* if(commentArray[4] == "z12bgp5ggnycy3i1p23hvnd4drfvx3tjw04#1423833570014115")
    for(var i=0; i < contentArray.length; i++){
      console.error(i);
      console.error(contentArray[i]);
    }
    */
  for(var i=0; i < contentArray.length; i++){
    var contentPart = contentArray[i];
    switch (contentPart[0]){
    case 1:
      contentStr += "<br/>";
      break;
    case 2:
      contentStr += '<a class="ot-anchor" href="' + contentPart[3][0] + '" target="_blank" rel="nofollow">' +  contentPart[1] + '</a>';
      break;
    case 3:
      contentStr += '<span class="proflinkWrapper"><span class="proflinkPrefix">+</span><a oid="' +
        contentPart[4][1] + '" href="/' + contentPart[4][1] + '" class="proflink">' + contentPart[1] + '</a></span>';
      break;
    default:
      contentStr += contentPart[1];
    }
  }
  var comment = {
    id: commentArray[4],
    author: {
      id: commentArray[6],
      name: commentArray[25][0],
      photo: commentArray[25][4]
    },
    content: contentStr,
    time: new Date(commentArray[3]),
    edited: commentArray[14] ? new Date(commentArray[14]) : null,
    plusone: {
      count: (typeof commentArray[15][12][0] != 'undefined') ? commentArray[15][12][0][0] : 0,
      isPlused: (commentArray[15][13] == '1'),
      id: commentArray[15][0] // 2015.12.14 @fronoske +1ユーザー表示のために plusoneId を取得しておく
    }
  };

  return comment;
};

/**
 * Parses a Google+ user array to object.
 *
 * @param {Array} userArray A Google+ user array.
 * @return {Object} A parsed user data object.
 */
GooglePlusAPI.prototype._parseUser = function (userArray) {
  var user = {
    id: userArray[0][2],
    name: userArray[2][0],
    photo: userArray[2][8],
    description: userArray[2][21]
  };

  return user;
};

/**
 * Parses a Google+ sharer array to object.
 *
 * @param {Array} sharerArray A Google+ sharer array.
 * @return {Object} A parsed sharer data object.
 */
GooglePlusAPI.prototype._parseSharer = function (sharerArray) {
  var sharer = {
    id: sharerArray[1],
    name: sharerArray[0],
    photo: sharerArray[4]
  };

  return sharer;
};

/**
 * Queries a specified post.
 *
 * @param {string} id A Google+ post ID.
 * @param {function (Object)} callback A callback.
 */
GooglePlusAPI.prototype.lookupPost = function (id, callback) {
  var self = this;
  var params = '?updateId=' + id;

  this._sendRequest(this.ACTIVITY_URL + params, null, function (response) {
    if (!self._isRequestSuccess(response, callback)) return;

    var post = self._parsePost(response[1]);
    self._fireCallback(true, post, callback);
  });
};

/**
 * Queries sharers of a specified post.
 *
 * @param {string} id A Google+ post ID.
 * @param {function (Object)} callback A callback.
 */
GooglePlusAPI.prototype.lookupSharers = function (id, callback) {
  var self = this;
  var params = '?id=' + id + '&rt=j';

  this._sendRequest(this.SHARERS_URL + params, null, function (response) {
    if (!self._isRequestSuccess(response, callback)) return;

    var sharers = [];
    response[1].forEach(function (element, index) {
      sharers.push(self._parseSharer(element));
    });

    self._fireCallback(true, sharers, callback);
  });
};

/**
 * Queries specified users.
 *
 * @param {(string|Array.<string>)} ids Google+ user IDs.
 * @param {function (Object)} callback A callback.
 */
GooglePlusAPI.prototype.lookupUsers = function (ids, callback) {
  var self = this;
  var allSlices = [];

  if (!Array.isArray(ids)) ids = [ids];
  ids.forEach(function (element, index) {
    allSlices.push('[null,null,"' + element + '"]');
  });

  var maxSlices = 12;
  var slicedIndex = 0;
  var doRequest = function () {
    var currentSlices = allSlices.slice(slicedIndex, slicedIndex + maxSlices);
    if (currentSlices.length === 0) return;
    slicedIndex += currentSlices.length;

    var params = '?n=6&m=[[' + currentSlices.join(',') + ']]&rt=j';
    var data = 'at=' + self._getSession();
    // console.log("[48CO] Lookup URL: " + self.LOOKUP_URL + params);
    // console.log("[48CO] postData=" + data);
    self._sendRequest(self.LOOKUP_URL + params, data, function (response) {
      if (!self._isRequestSuccess(response, callback)) {
        doRequest();
        return;
      }

      var users = {};

      response[0][1].forEach(function (element, index) {
        var user = self._parseUser(element[1]);
        users[user.id] = user;
      });
      self._fireCallback(true, users, callback);
      doRequest();
    });
  };

  doRequest();
};

/**
 * Queries Google+ current login user data.
 *
 * @param {function (Object)} callback A callback.
 */
GooglePlusAPI.prototype.lookupCurrentUser = function (callback) {
  var self = this;
  if (!this._getSession()) {
    self._fireCallback(false, 'Not Login', callback);
    return;
  }

  this._sendRequest(this.INITIAL_DATA_URL + '&rt=j', null, function (response) {
    if (!self._isRequestSuccess(response, callback)) return;

    var info = self._parseJSON(response[0][1]);
    var data = {};

    for (var i in info) {
      data.id = info[i][0];
      data.name = info[i][20].replace(/(.+) <(.+)>/, '$1');
      data.photo = info[i][1][3];
      break;
    }

    self._fireCallback(true, data, callback);
  });
};

/**
 * Edits a specified comment posted by current login user.
 *
 * @param {string} id Google+ comment ID.
 * @param {string} text Edited comment content.
 * @param {function (Object)} callback A callback.
 */
GooglePlusAPI.prototype.editComment = function (id, text, callback) {
  var self = this;
  var postID = id.split('#')[0];
  var data = 'at=' + this._getSession() +
    '&itemId=' + postID +
    '&commentId=' + id +
    '&text=' + encodeURIComponent(text);
  this._sendRequest(this.COMMENT_EDIT_URL, data, function (response) {
    if (!self._isRequestSuccess(response, callback)) return;
    var comment = self._parseComment(response[1]);
    self._fireCallback(true, comment, callback);
  });
};

/**
 * Deletes a specified comment posted by current login user.
 *
 * @param {string} id Google+ comment ID.
 * @param {function (Object)} callback A callback.
 */
GooglePlusAPI.prototype.deleteComment = function (id, callback) {
  var self = this;
  var data = 'at=' + this._getSession() +
 '&commentId=' + id;

  this._sendRequest(this.COMMENT_DELETE_URL, data, function (response) {
    if (!self._isRequestSuccess(response, callback)) return;
    self._fireCallback(true, null, callback);
  });
};

/**
 * +1 a specified comment.
 */
GooglePlusAPI.prototype.plusOneComment = function (id, set, callback) {
  var self = this;
  var data = 'at=' + this._getSession() +
    '&itemId=' + encodeURIComponent('comment:' + id) +
    '&set=' + (set ? 'true' : 'false');

  this._sendRequest(this.COMMENT_PLUSONE_URL, data, function (response) {
    if (!self._isRequestSuccess(response, callback)) return;
    self._fireCallback(true, null, callback);
  });
};

/**
 * +1 user list (2015.12.14 @fronoske)
 */
GooglePlusAPI.prototype.plusOneUserList = function (id, callback) {
  var self = this;
  var data = 'at=' + this._getSession() +
    '&plusoneId=' + encodeURIComponent(id) +
    '&num=999';
  
  this._sendRequest(this.PLUSONE_USERS_URL, data, function (response) {
    if (!self._isRequestSuccess(response, callback)) return;
    self._fireCallback(true, response, callback);
  });
};


Date.prototype.toShortString = function () {
  var month = this.getMonth() + 1;
  var date = this.getDate();
  var hours = this.getHours();
  var minutes = (this.getMinutes() < 10) ? '0' + this.getMinutes() : this.getMinutes();

  return month + '/' + date + ' ' + hours + ':' + minutes;
};

Date.prototype.toLongString = function () {
  var year = this.getFullYear();
  var month = this.getMonth() + 1;
  var date = this.getDate();
  var hours = (this.getHours() < 10) ? '0' + this.getHours() : this.getHours();
  var minutes = (this.getMinutes() < 10) ? '0' + this.getMinutes() : this.getMinutes();
  var seconds = (this.getSeconds() < 10) ? '0' + this.getSeconds() : this.getSeconds();

  return year + '/' + month + '/' + date + ' ' + hours + ':' + minutes + ':' + seconds;
};

Date.prototype.toHumanTimeDiff = function (format) {
  var now = new Date();
  var diff = (now.getTime() - this.getTime()) / 1000;
  var string = '';

  if (diff < 60) {
    return format[0];
  }
  if (diff < 3600) {
    diff = Math.round(diff / 60);
    string = (diff == 1) ? format[1] : format[2];
    return string.replace('%', diff);
  }
  if (diff < 86400) {
    diff = Math.round(diff / 3600);
    string = (diff == 1) ? format[3] : format[4];
    return string.replace('%', diff);
  }

  diff = Math.round(diff / 86400);
  string = (diff == 1) ? format[5] : format[6];
  return string.replace('%', diff);
};

new CommentsOnly();

if (typeof chrome === 'object') {
  chrome.runtime.onMessage.addListener(function (request, sender, sendResponse) {
    if (request.action === 'pageAction') {
      $('.ext48co-button-settings').eq(0).click();
    }
  });
}