- // ==UserScript==
- // @name Append Tag Searching Tub
- // @name:ja niconico タグ検索タブを追加
- // @description Adds “Keyword”, “Tags”, “My List”, “Images” and “Live” search tabs to all of the Niconico search boxes.
- // @description:ja 『ニコニコ』各サービスの検索窓について、「キーワード」「タグ」「マイリスト」「静画」「生放送」検索タブが5つとも含まれるように補完します。
- // @namespace http://loda.jp/script/
- // @version 5.3.0
- // @match https://www.nicovideo.jp/
- // @match https://www.nicovideo.jp/?*
- // @match https://www.nicovideo.jp/#*
- // @match https://www.nicovideo.jp/tag/*
- // @match https://www.nicovideo.jp/related_tag/*
- // @match https://www.nicovideo.jp/mylist*
- // @match https://www.nicovideo.jp/search/*
- // @match https://seiga.nicovideo.jp/*
- // @match https://live.nicovideo.jp/*
- // @match https://com.nicovideo.jp/*
- // @match *://blog.nicovideo.jp/en_info/*
- // @match *://tw.blog.nicovideo.jp/*
- // @require https://gitcdn.xyz/cdn/greasemonkey/gm4-polyfill/a834d46afcc7d6f6297829876423f58bb14a0d97/gm4-polyfill.js
- // @require https://greasyfork.org/scripts/19616/code/utilities.js?version=868689
- // @license MPL-2.0
- // @contributionURL https://www.amazon.co.jp/registry/wishlist/E7PJ5C3K7AM2
- // @compatible Edge
- // @compatible Firefox 推奨 / Recommended
- // @compatible Opera
- // @compatible Chrome
- // @grant GM.setValue
- // @grant GM_setValue
- // @grant GM.getValue
- // @grant GM_getValue
- // @grant GM.deleteValue
- // @grant GM_deleteValue
- // @grant GM.xmlHttpRequest
- // @grant GM_xmlhttpRequest
- // @connect www.nicovideo.jp
- // @run-at document-start
- // @icon https://nicovideo.cdn.nimg.jp/uni/images/favicon/144.png
- // @author 100の人
- // @homepageURL https://greasyfork.org/scripts/268
- // ==/UserScript==
-
- 'use strict';
-
- // L10N
- Gettext.setLocalizedTexts({
- /*eslint-disable quote-props, max-len */
- 'en': {
- 'キーワード': 'Keyword',
- '動画をキーワードで検索': 'Search Video by Keyword',
- 'タグ': 'Tags',
- '動画をタグで検索': 'Search Video by Tag',
- 'マイリスト': 'My List',
- 'マイリストを検索': 'Search My List',
- '静画': 'Images',
- '静画を検索': 'Search Images',
- '生放送': 'Live',
- '番組を探す': 'Search Live Program',
- 'マンガ': 'Comics',
- },
- 'zh': {
- 'キーワード': '關鍵字',
- '動画をキーワードで検索': '',
- 'タグ': '標籤',
- '動画をタグで検索': '',
- 'マイリスト': '我的清單',
- 'マイリストを検索': '搜尋我的清單',
- '静画': '靜畫',
- '静画を検索': '搜尋靜畫',
- '生放送': '生放送',
- '番組を探す': '搜尋節目',
- 'マンガ': '漫畫',
- },
- /*eslint-enable quote-props, max-len */
- });
-
-
-
- /**
- * 追加したタブバーから新しいタブで検索結果を開いたとき、選択中のタブを元に戻す遅延時間 (ミリ秒)。
- * @constant {number}
- */
- const CURRENT_TAB_RESTORATION_DELAY = 1000;
-
- /**
- * 表示しているページの種類。
- * @type {string}
- */
- let pageType;
-
- // ページの種類を取得
- switch (location.host) {
- case 'www.nicovideo.jp':
- if (location.pathname === '/') {
- // 総合トップページ
- pageType = 'top';
- } else if (location.pathname.startsWith('/search/')) {
- // 動画キーワード検索ページ
- pageType = 'videoSearch';
- } else if (location.pathname.startsWith('/mylist_search')) {
- // マイリスト検索ページ
- pageType = 'mylist';
- } else if (/^\/(?:(?:tag|related_tag)\/|(?:mylist|recent|newarrival|openlist|video_catalog)(?:\/|$))/
- .test(location.pathname)) {
- // 動画タグ検索ページと公開マイリスト等
- pageType = 'tag';
- } else if (location.pathname.startsWith('/user/')) {
- // ユーザーページ
- pageType = 'user';
- }
- break;
- case 'seiga.nicovideo.jp':
- pageType = location.pathname.startsWith('/search/')
- // 静画検索ページ
- ? 'imageSearch'
- // 静画ページ
- : 'image';
- break;
- case 'live.nicovideo.jp':
- pageType = location.pathname.startsWith('/search')
- // 生放送検索ページ
- ? 'liveSearch'
- // 生放送ページ
- : 'live';
- break;
- case 'blog.nicovideo.jp':
- // 英語版ニコニコインフォ
- pageType = 'info_en';
- break;
- case 'tw.blog.nicovideo.jp':
- // 台湾版ニコニコインフォ
- pageType = 'info_tw';
- break;
- }
-
- waitTarget(() => document.documentElement).then(function () {
- Gettext.setLocale(document.documentElement.lang);
- });
-
- if (pageType.startsWith('info_')) {
- // 英語版、または台湾版のニコニコインフォなら
- waitTarget(() => document.getElementById('siteHeaderLeftMenu')).then(function () {
- // 生放送へのリンクを取得
- const itemLive = document.querySelector('#siteHeader [href*="://live.nicovideo.jp/"]').parentElement;
- // 生放送リンクの複製
- const item = itemLive.cloneNode(true);
- // リンク文字を変更
- item.getElementsByTagName('span')[0].textContent = _('静画');
- // アドレスを変更
- item.getElementsByTagName('a')[0].href = 'https://seiga.nicovideo.jp/';
- // ヘッダに静画へのリンクを追加
- itemLive.before(item);
- });
- } else {
- // ページの種類別に、実行する関数を切り替える。
- switch (pageType) {
- case 'videoSearch': // 動画キーワード
- case 'mylist': // マイリスト
- waitTarget(() => document.getElementById('search_united_form')).then(addTagSearchTabAboveSearchBox);
- break;
-
- case 'top':
- // トップページ
- addTagSearchButtonToTopPage();
- break;
-
- case 'imageSearch':
- // 静画キーワード
- waitTarget(() => document.getElementById('usearch_form_input')).then(addTagSearchTabAboveSearchBox);
- break;
-
- case 'image':
- // 静画
- waitTarget(() => document.getElementById('search_button')).then(careteTabsBarToSearchBox);
- break;
-
- case 'liveSearch': {
- // 生放送キーワード
- const forms = document.getElementsByClassName('search-form');
- waitTarget(() => forms[0]).then(addTagSearchTabAboveSearchBox);
- break;
- }
- case 'live': {
- // 生放送
- const words = document.getElementsByClassName('search_word');
- waitTarget(() => words[0]).then(careteTabsBarToSearchBox);
- break;
- }
-
- case 'tag':
- if (document.doctype.publicId) {
- // 公開マイリスト等
- waitTarget(() => document.getElementById('target_m')).then(addOtherServiceTabsAboveSearchBox);
- } else {
- // 動画タグ
- const mylists = document.getElementsByClassName('optMylist');
- waitTarget(() => mylists[0]).then(addOtherServiceTabsAboveSearchBox);
- }
- break;
-
- case 'user': {
- // ユーザー
- const outers = document.getElementsByClassName('optionOuter');
- waitTarget(() => outers[0]).then(addImageLinkToUserPageMenu);
- break;
- }
- }
- }
-
-
-
-
- /**
- * 各サービスのキーワード検索ページの検索窓に、動画の「タグ」検索タブを追加する。
- */
- function addTagSearchTabAboveSearchBox()
- {
- // マイリスト検索タブの取得
- const mylistTab = document.querySelector('.tab_table td:nth-of-type(2), #search_frm_a a:nth-of-type(2), .search_tab_list li:nth-of-type(2), .seachFormA a:nth-of-type(2), li:nth-of-type(2).search-tab-item');
-
- // マイリスト検索タブの複製
- const tagTab = mylistTab.cloneNode(true);
-
- // タブ名を変更
- const anchor = tagTab.tagName.toLowerCase() === 'a' ? tagTab : tagTab.getElementsByTagName('a')[0];
- let tabNameNode = anchor.getElementsByTagName('div');
- tabNameNode = (tabNameNode.length > 0 ? tabNameNode[0].firstChild : anchor.firstChild);
- tabNameNode.data = _('タグ') + (pageType === 'liveSearch' ? '' : ' ( ');
-
- // クラス名を変更・動画件数をリセット
- const searchCount = tagTab.querySelector('strong, span');
- switch (pageType) {
- case 'videoSearch':
- searchCount.classList.remove('more');
- break;
- case 'mylist':
- searchCount.style.removeProperty('color');
- break;
- case 'imageSearch':
- searchCount.classList.remove('search_value_em');
- searchCount.classList.add('search_value');
- break;
- }
- searchCount.textContent = '-';
-
- if (searchCount.id) {
- // 生放送
- searchCount.id = 'search_count_tag';
- }
-
- // 検索語句を取得
- const searchWordsPattern = /(?:\/(?:search|tag|mylist_search)\/|[?&]keyword=)([^?&#]+)/g;
- const result = location.href.match(searchWordsPattern);
- const searchWords
- = result ? searchWordsPattern.exec(result[pageType === 'liveSearch' ? result.length - 1 : 0])[1] : '';
-
- // タグが付いた動画件数を取得・表示
- if (searchWords && location.host !== 'www.live.nicovideo.jp') {
- GM.xmlHttpRequest({
- method: 'GET',
- url: 'https://www.nicovideo.jp/tag/' + searchWords,
- onload: function (response) {
- const responseDocument = new DOMParser().parseFromString(response.responseText, 'text/html');
- const total = responseDocument.querySelector('.tagCaption .dataValue .num').textContent;
-
- const trimmedThousandsSep = total.replace(/,/g, '');
- if (trimmedThousandsSep >= 100) {
- // 動画件数が100件を超えていれば
- switch (pageType) {
- case 'videoSearch':
- searchCount.classList.add('more');
- break;
- case 'mylist':
- searchCount.style.color = '#CC0000';
- break;
- case 'imageSearch':
- searchCount.classList.remove('search_value');
- searchCount.classList.add('search_value_em');
- break;
- case 'liveSearch':
- searchCount.classList.add('strong');
- break;
- }
- }
-
- switch (pageType) {
- case 'mylist':
- searchCount.textContent = ' ' + total + ' ';
- break;
- case 'videoSearch':
- case 'imageSearch':
- searchCount.textContent = total;
- break;
- case 'liveSearch':
- searchCount.textContent = trimmedThousandsSep;
- break;
- }
- },
- });
- }
-
- // 非アクティブタブを取得
- const inactiveTab = document.querySelector('.tab_0, .tab1, .search_tab_list a:not(.active), .search-tab-anchor');
-
- // クラス名を変更
- anchor.className = inactiveTab.className;
-
- // アドレスを変更
- anchor.href = 'https://www.nicovideo.jp/tag/' + searchWords + inactiveTab.search;
-
- // タグ検索タブを追加
- mylistTab.parentNode.insertBefore(tagTab, mylistTab);
- if (pageType === 'liveSearch') {
- mylistTab.parentNode.insertBefore(new Text(' '), mylistTab);
- } else if (inactiveTab.classList.contains('tab1')) {
- // GINZAバージョン
- mylistTab.parentNode.insertBefore(tagTab.previousSibling.cloneNode(true), mylistTab);
- }
- }
-
-
-
- /**
- * ニコニコ動画の上部に表示されている検索窓に、「静画」「生放送」を検索するタブを追加する。
- */
- function addOtherServiceTabsAboveSearchBox()
- {
- // スタイルの設定
- document.head.insertAdjacentHTML('beforeend', `<style>
- :root {
- --max-search-box-width: 268px;
- }
- #PAGEHEADER > div {
- display: flex;
- }
- #head_search {
- max-width: var(--max-search-box-width);
- flex-grow: 1;
- }
- #search_input {
- width: 100%;
- display: flex;
- }
- #search_input .typeText {
- flex-grow: 1;
- }
- #head_ads {
- margin-right: -26px;
- }
- #search_input #bar_search {
- box-sizing: border-box;
- width: 100% !important;
- }
- /*====================================
- GINZAバージョン
- */
- .siteHeader > .inner {
- display: flex;
- }
- .videoSearch {
- max-width: var(--max-search-box-width);
- flex-grow: 1;
- padding-left: 4px;
- padding-right: 4px;
- }
- .videoSearchOption {
- display: flex;
- white-space: nowrap;
- }
- .videoSearch form {
- display: flex;
- }
- .videoSearch form .inputText {
- flex-grow: 1;
- }
- /*------------------------------------
- ×ボタン
- */
- .clear-button-inner-tag {
- left: initial;
- right: 3px;
- }
- </style>`);
-
- // タブリストの取得
- const mylistTab = document.querySelector('#target_t, .optMylist');
-
- // タブの複製・追加
- mylistTab.parentElement.append(...[
- {
- type: 'image',
- title: _('静画を検索'),
- url: 'https://seiga.nicovideo.jp/search',
- text: _('静画'),
- },
- {
- type: 'live',
- title: _('番組を探す'),
- url: 'https://live.nicovideo.jp/search',
- text: _('生放送'),
- },
- ].map(function (option) {
- const tab = mylistTab.cloneNode(true);
- if (mylistTab.classList.contains('optMylist')) {
- // GINZAバージョン
- tab.classList.remove('optMylist');
- tab.classList.add('opt' + option.type[0].toUpperCase() + option.type.slice(1));
- tab.dataset.type = option.type;
- tab.getElementsByTagName('a')[0].textContent = option.text;
- } else {
- // 公開マイリスト等
- tab.id = 'target_' + option.type[0];
- tab.title = option.title;
- tab.setAttribute('onclick', tab.getAttribute('onclick').replace(/'.+?'/, '\'' + option.url + '\''));
- tab.textContent = option.text;
- }
- return tab;
- }));
-
- GreasemonkeyUtils.executeOnUnsafeContext(/* global Nico */ function () {
- eval('Nico.Navigation.HeaderSearch.Controller.search = '
- + Nico.Navigation.HeaderSearch.Controller.search.toString().replace(/(switch.+?{[^}]+)/, `$1;
- break;
- case "image":
- d = "https://seiga.nicovideo.jp/search/" + e;
- break;
- case "live":
- d = "https://live.nicovideo.jp/search/" + e;
- break;
- `));
- });
- }
-
-
-
- /**
- * 静画・生放送の上部に表示されている検索窓に、「動画キーワード」「動画タグ」「マイリスト」「静画」「生放送」を検索するタブバーを設置する。
- */
- function careteTabsBarToSearchBox()
- {
- // スタイルの設定
- document.head.insertAdjacentHTML('beforeend', `<style>
- #sg_search_box {
- /* 静画 */
- margin-top: 0.2em;
- }
- #live_header div.score_search { /* 生放送マイページ向けに詳細度を大きくしている */
- /* 生放送 */
- top: initial;
- }
- /*------------------------------------
- タブバー
- */
- [action$="search"] > ul {
- display: flex;
- /* 生放送 */
- font-size: 12px;
- }
- /* 静画 */
- #head_search_form > ul {
- margin-left: 1.3em;
- /* マンガ・電子書籍 */
- line-height: 1.4em;
- }
- #head_search_form > ul:hover ~ .search_form_text {
- border-color: #999;
- }
- /*------------------------------------
- タブ
- */
- [action$="search"] > ul > li {
- margin-left: 0.2em;
- white-space: nowrap;
- }
- [action$="search"] > ul > li > a {
- background: lightgrey;
- padding: 0.2em 0.3em 0.1em;
- color: inherit;
- /* 生放送 */
- text-decoration: none;
- }
- #head_search_form > ul > li > a:hover {
- /* 静画 */
- text-decoration: none;
- }
- /*------------------------------------
- 選択中のタブ
- */
- [action$="search"] > ul > li.current > a {
- color: white;
- background: dimgray;
- }
- </style>`);
-
- /**
- * 静画検索のtargetパラメータの値。
- * @type {string}
- */
- let imageSearchParamValue = 'illust';
-
- const form = document.querySelector('[action$="search"]');
- const textField = form[pageType === 'image' ? 'q' : 'keyword'];
-
- if (pageType === 'image') {
- // 静画の場合
- const pathnameParts = document.querySelector('#logo > h1 > a').pathname.split('/');
- switch (pathnameParts[1]) {
- case 'manga':
- imageSearchParamValue = 'manga';
- break;
- case 'book':
- imageSearchParamValue = pathnameParts[2] === 'r18' ? 'book_r18' : 'book';
- break;
- }
- }
-
- form.insertAdjacentHTML('afterbegin', `<ul>
- <li>
- <a href="https://www.nicovideo.jp/search/" title="${h(_('動画をキーワードで検索'))}">${h(_('キーワード'))}</a>
- </li>
- <li>
- <a href="https://www.nicovideo.jp/tag/" title="${h(_('動画をタグで検索'))}">${h(_('タグ'))}</a>
- </li>
- <li>
- <a href="https://www.nicovideo.jp/mylist_search/" title="${h(_('マイリストを検索'))}">${h(_('マイリスト'))}</a>
- </li>
- <li${pageType === 'image' ? ' class="current"' : ''}>
- <a href="https://seiga.nicovideo.jp/search/?target=${imageSearchParamValue}"
- title="${h(textField.defaultValue)}">${h(_('静画'))}</a>
- </li>
- <li${pageType === 'live' ? ' class="current"' : ''}>
- <a href="https://live.nicovideo.jp/search/" title="' + h(_('番組を探す')) + '">${h(_('生放送'))}</a>
- </li>
- </ul>`);
-
- const defaultCurrentTabAnchor = form.querySelector('.current a');
-
- document.addEventListener('click', function (event) {
- if (event.button !== 2 && event.target.matches('[action$="search"] > ul > li > a')) {
- // タブが副ボタン以外でクリックされたとき
- let searchWord = textField.value.trim();
- if (pageType === 'image' && textField.value === textField.defaultValue) {
- // 静画の場合、検索窓の値が既定値と一致していれば空欄とみなす
- searchWord = '';
- }
- if (searchWord) {
- // 検索語句が入力されていれば
- switchTab(event.target);
- event.target.pathname = event.target.pathname.replace(/[^/]*$/, encodeURIComponent(searchWord));
- setTimeout(function () {
- // リンク先を新しいタブで開いたとき
- switchTab(defaultCurrentTabAnchor);
- }, CURRENT_TAB_RESTORATION_DELAY);
- } else {
- // 検索語句が未入力なら
- event.preventDefault();
- if (event.button === 0) {
- // 主ボタンでクリックされていれば
- switchTab(event.target);
- }
- }
- }
- });
-
- // TabSubmitをインストールしているとマウスボタンを取得できず、中クリック時にも同じタブで検索してしまうため分割
- form.addEventListener('click', function (event) {
- if (event.target.type === (pageType === 'image' ? 'image' : 'submit')) {
- // 送信ボタンをクリックしたとき
- const searchWord = textField.value !== textField.defaultValue && textField.value.trim();
- if (searchWord) {
- event.stopPropagation();
- event.preventDefault();
- const anchor = form.querySelector('.current a');
- anchor.pathname = anchor.pathname.replace(/[^/]*$/, encodeURIComponent(searchWord));
- location.assign(anchor.href);
- }
- }
- }, true);
-
- addEventListener('pageshow', function (event) {
- if (event.persisted) {
- // 履歴にキャッシュされたページを再表示したとき
- switchTab(defaultCurrentTabAnchor);
- }
- });
-
- /**
- * 選択しているタブを切り替える。
- * @param {HTMLAnchorElement} target - 切り替え先のタブのリンク。
- */
- function switchTab(target) {
- form.getElementsByClassName('current')[0].classList.remove('current');
- target.parentElement.classList.add('current');
- if (pageType === 'image') {
- // 静画
- if (textField.defaultValue === textField.value) {
- // 検索語句が未入力なら
- textField.defaultValue = textField.value = target.title;
- } else {
- // 検索語句が入力されていれば
- textField.defaultValue = target.title;
- }
- } else {
- // 生放送
- textField.placeholder = target.title;
- }
- }
- }
-
-
-
- /**
- * 総合トップページの検索窓に、動画「タブ」「マイリスト」検索ボタンを追加する。
- */
- function addTagSearchButtonToTopPage()
- {
- // スタイルの設定
- document.head.insertAdjacentHTML('beforeend', `<style>
- .CrossSearch {
- display: flex;
- margin-right: 1em;
- }
- .CrossSearch-services {
- display: flex;
- }
- .CrossSearch-service {
- width: unset;
- padding: 0 0.5em;
- white-space: nowrap;
- }
- .CrossSearch-form {
- width: unset;
- }
- </style>`);
-
- // 静画検索ボタンの取得
- const refItem = document.querySelector('.CrossSearch-service[data-service="seiga"]');
-
- const tagItem = refItem.cloneNode(true);
- tagItem.textContent = _('タグ');
- tagItem.dataset.service = 'tag';
- tagItem.dataset.baseUrl = 'https://www.nicovideo.jp/tag/';
- refItem.before(tagItem);
-
- const mylist = refItem.cloneNode(true);
- mylist.textContent = _('マイリスト');
- mylist.dataset.service = 'mylist';
- mylist.dataset.baseUrl = 'https://www.nicovideo.jp/mylist_search/';
- refItem.before(mylist);
- }
-
-
-
- /**
- * ユーザーページ左側のメニューに、静画へのリンクを追加する。
- */
- function addImageLinkToUserPageMenu()
- {
- // スタイルの設定
- document.head.insertAdjacentHTML('beforeend', `<style>
- .sidebar ul li.imageTab a span {
- width: 22px;
- height: 20px;
- background: url("");
- }
- </style>`);
-
- const nextItem = document.getElementsByClassName('stampTab')[0];
-
- const item = nextItem.cloneNode(true);
- const classList = item.classList;
- classList.remove('stampTab', 'active');
- classList.add('imageTab');
- const anchor = item.getElementsByTagName('a')[0];
- anchor.href = 'https://seiga.nicovideo.jp/user/illust/' + /[0-9]+/.exec(anchor.pathname)[0];
- anchor.lastChild.data = _('静画');
-
- nextItem.prepend(item);
- }