Greasy Fork is available in English.

pixiv 検索オプションを追加

This is a search option add-on script for pixiv. Enables you to select search mode by radio buttons (legacy feature).

ของเมื่อวันที่ 30-07-2015 ดู เวอร์ชันล่าสุด

// ==UserScript==
// @name        pixiv 検索オプションを追加
// @name:ja     pixiv 検索オプションを追加
// @namespace   http://loda.jp/script/
// @id          pixiv-search-box-347021
// @version     2.6.0
// @description This is a search option add-on script for pixiv. Enables you to select search mode by radio buttons (legacy feature).
// @description:ja 検索オプションを、以前のようにラジオボタンで選択出来るようにする。
// @match       http://www.pixiv.net/*
// @exclude     http://www.pixiv.net/member_illust.php?*mode=manga*
// @exclude     http://www.pixiv.net/member_illust.php?*mode=big*
// @exclude     http://www.pixiv.net/apps.php*
// @domain      www.pixiv.net
// @match       https://booth.pm/*
// @exclude     https://booth.pm/users/*
// @domain      booth.pm
// @run-at      document-start
// @grant       dummy
// @icon        
// @author      100の人
// @homepage    https://greasyfork.org/scripts/265
// @license     Creative Commons Attribution 4.0 International Public License; http://creativecommons.org/licenses/by/4.0/
// ==/UserScript==

(function () {
'use strict';

polyfill();

// L10N
Gettext.setLocalizedTexts({
	'en': {
		'完全一致': 'Exact match',
		'部分一致': 'Partial match',
		'タイトル・キャプション': 'Title/Description',
		'小説': 'Novels',
		'タグ': 'Tags',
		'キーワード': 'Keyword',
		'本文': 'Content',
		'ユーザー': 'User',
		'グループ': 'Groups',
		'BOOTH': '',
		'すべて': 'All',
		'検索': 'Search',
		'小説検索': 'Search novel',
		'ユーザー検索': 'Search user',
		'グループ検索': 'Search group',
		'商品検索...': 'Search',
		'https://booth.pm/ja/search/': 'https://booth.pm/en/search/',
	},
	'fr': {
		'完全一致': 'Concordance parfaite',
		'部分一致': 'Concordance partielle',
		'タイトル・キャプション': 'Titre, Légende',
		'小説': 'Roman',
		'タグ': 'Mots-clés',
		'キーワード': 'Mots-clés',
		'本文': 'Contenu',
		'ユーザー': 'Utilisateur',
		'グループ': 'Groups',
		'BOOTH': '',
		'すべて': 'Tout',
		'検索': 'Rechercher',
		'小説検索': 'Rechercher un roman',
		'ユーザー検索': '',
		'グループ検索': 'Rechercher un groupe',
		'商品検索...': '',
		'https://booth.pm/ja/search/': '',
	},
	'ko': {
		'完全一致': '완전 일치',
		'部分一致': '부분 일치',
		'タイトル・キャプション': '제목・캡션',
		'小説': '소설',
		'タグ': '태그',
		'キーワード': '키워드',
		'本文': '본문',
		'ユーザー': '유저',
		'BOOTH': '',
		'グループ': '그룹',
		'すべて': '전체',
		'検索': '검색',
		'小説検索': '소설 검색',
		'ユーザー検索': '유저 검색',
		'グループ検索': '그룹 검색',
		'商品検索...': '',
		'https://booth.pm/ja/search/': '',
	},
	'ru': {
		'完全一致': 'Полное совпадение',
		'部分一致': 'Частичное совпадение',
		'タイトル・キャプション': 'Заголовок',
		'小説': 'Рассказы',
		'タグ': 'Метка',
		'キーワード': 'Ключевые слова',
		'本文': 'Текст',
		'ユーザー': 'Пользователь',
		'グループ': 'Группа',
		'BOOTH': '',
		'すべて': 'Все',
		'検索': 'Поиск',
		'小説検索': 'Искать рассказ',
		'ユーザー検索': 'Искать пользователя',
		'グループ検索': 'Искать группу',
		'商品検索...': '',
		'https://booth.pm/ja/search/': '',
	},
	'th': {
		'完全一致': '',
		'部分一致': '',
		'タイトル・キャプション': 'ชื่อและคำบรรยาย',
		'小説': 'นิยาย',
		'タグ': 'แท็ก',
		'キーワード': 'คีย์เวิร์ด',
		'本文': '',
		'ユーザー': 'ผู้ใช้',
		'グループ': '',
		'BOOTH': '',
		'すべて': 'ทั้งหมด',
		'検索': 'ค้นหา',
		'小説検索': '',
		'ユーザー検索': '',
		'グループ検索': '',
		'商品検索...': '',
		'https://booth.pm/ja/search/': '',
	},
	'zh': {
		'完全一致': '完全相同',
		'部分一致': '部分相同',
		'タイトル・キャプション': '题目/简述',
		'小説': '小说',
		'タグ': '标签',
		'キーワード': '关键词',
		'本文': '内容',
		'ユーザー': '用户',
		'グループ': '群组',
		'BOOTH': '',
		'すべて': '全部',
		'検索': '搜索',
		'小説検索': '搜索小说',
		'ユーザー検索': '搜索用户',
		'グループ検索': '搜索群组',
		'商品検索...': '',
		'https://booth.pm/ja/search/': '',
	},
	'zh-tw': {
		'完全一致': '完全相同',
		'部分一致': '部分相同',
		'タイトル・キャプション': '題目/簡述',
		'小説': '小說',
		'タグ': '標籤',
		'キーワード': '關鍵詞',
		'本文': '內容',
		'ユーザー': '用戶',
		'グループ': '群組',
		'BOOTH': '',
		'すべて': '全部',
		'検索': '搜索',
		'小説検索': '搜索小說',
		'ユーザー検索': '搜索用戶',
		'グループ検索': '搜索群組',
		'商品検索...': '',
		'https://booth.pm/ja/search/': '',
	},
	'es': {
		'完全一致': 'Coincidencia exacta',
		'部分一致': 'Coincidencia parcial',
		'タイトル・キャプション': 'Título/Descripción',
		'小説': 'Novelas',
		'タグ': 'Etiquetas',
		'キーワード': 'Palabra clave',
		'本文': 'Mensaje',
		'ユーザー': 'Usuarios',
		'グループ': 'Grupo',
		'BOOTH': '',
		'すべて': 'Todos',
		'検索': 'Buscar',
		'小説検索': 'Buscar novela',
		'ユーザー検索': 'Buscar usuario',
		'グループ検索': 'Buscar grupo',
		'商品検索...': '',
		'https://booth.pm/ja/search/': '',
	},
});



if (window.location.host === 'booth.pm') {
	startScript(main,
			function (parent) { return parent.localName === 'nav'; },
			function (target) { return target.classList.contains('ctrl-nav'); },
			function () { return document.getElementsByClassName('ctrl-nav')[0]; }, {
				isTargetParent: function (parent) { return parent.localName === 'html'; },
				isTarget: function (target) { return target.localName === 'body'; },
			});
} else {
	startScript(main,
			function (parent) { return parent.localName === 'body'; },
			function (target) { return target.id === 'wrapper'; },
			function () { return document.getElementById('wrapper'); });
}

function main() {
	var loggedin = !document.body.classList.contains('not-logged-in'), booth = window.location.host === 'booth.pm',
			form, submitButton, word = document.getElementById(booth ? 'query' : 'suggest-input'), mode, subMode, tabs,
			input, searchParams;

	// 言語の設定
	Gettext.setLocale(booth ? window.navigator.language : document.documentElement.lang);

	// スタイルシートの設定
	document.head.insertAdjacentHTML('beforeend', '<style> \
		.ui-search, \
		.global-nav .item-search-box .item-search { \
			display: flex; \
			align-items: center; \
			width: initial; \
			/* BOOTH */ \
			max-width: initial; \
		} \
		.ui-search { \
			background: #FFFFFF; \
			padding: 0.4em 2em; \
			border-left: 1px solid #D6DEE5; \
			border-right: 1px solid #D6DEE5; \
			border-bottom: 1px solid #D6DEE5; \
			border-radius: 0 0 5px 5px; \
			justify-content: center; \
			/* 通知が検索窓に隠れる問題の修正 */ \
			z-index: 98; \
			/* トップページ・pixivについて */ \
			margin-bottom: 10px; \
		} \
		.not-logged-in .ui-search { \
			/* ログイン前 */ \
			border-top: initial; \
			border-top-left-radius: initial; \
			border-top-right-radius: initial; \
		} \
		\
		/*------------------------------------ \
			検索窓 \
		*/ \
		.ui-search .container, \
		.global-nav .item-search-box .twitter-typeahead { \
			flex-grow: 1; \
			flex-shrink: 0; \
			width: initial; \
			max-width: 250px; \
		} \
		.ui-search div.container { \
			border-radius: 0; \
		} \
		.global-nav .item-search-box .twitter-typeahead { \
			max-width: 356px; \
		} \
		#suggest-input { \
			width: 100%; \
		} \
		\
		/*------------------------------------ \
			送信ボタン \
		*/ \
		.ui-search input.submit { \
			position: static; \
			margin-right: 1em; \
		} \
		\
		/*------------------------------------ \
			検索オプション \
		*/ \
		.ui-search label, \
		.item-search label { \
			padding: 0 0.7em; \
			display: flex; \
			align-items: center; \
			white-space: nowrap; \
		} \
		.ui-search label input { \
			margin-right: 0.3em; \
			width: initial; \
		} \
		.item-search label, \
		.item-search .sub-options { \
			/* BOOTH */ \
			font-size: 12px; \
			font-weight: normal; \
		} \
		\
		/*------------------------------------ \
			副検索モード \
		*/ \
		.sub-options label:not(:first-of-type) { \
			display: none; \
			position: absolute; \
			z-index: 1; \
			width: 13em; \
			height: 2em; \
			border: solid 1px #D6DEE5; \
			border-top: none; \
			border-bottom: none; \
			background: #FFFFFF; \
		} \
		.sub-options::after { \
			content: ""; \
			padding: 0 0.7em; \
			display: none; \
			position: absolute; \
			z-index: 1; \
			width: 13em; \
			height: 17px; \
			border: solid 1px #FFFFFF; \
			border-top: none; \
			border-bottom: none; \
			bottom: 0; \
		} \
		.item-search .sub-options::after { \
			/* BOOTH */ \
			top: 23px; \
			bottom: initial; \
		} \
		.sub-options label:nth-of-type(3) { \
			margin-top: 2em; \
		} \
		.sub-options label:nth-of-type(4) { \
			margin-top: 4em; \
		} \
		.sub-options label:nth-of-type(5) { \
			margin-top: 6em; \
		} \
		.sub-options label:last-of-type { \
			border-bottom: 1px solid #D6DEE5; \
			border-radius: 0 0 5px 5px; \
		} \
		.sub-options:hover label { \
			display: flex; \
		} \
		.sub-options:hover::after { \
			display: block; \
		} \
		\
		/*------------------------------------ \
			副検索モードが右側にはみ出す問題に対処 \
		*/ \
		.ctrl-nav { \
			/* お買い物ガイド */ \
			position: relative; \
			z-index: 1; \
		} \
		.ctrl-nav::after { \
			content: ""; \
			position: absolute; \
			width: 100%; \
			border-bottom: solid 1px #D3D3D3; \
			left: -1px; \
			top: 48px; \
		} \
		\
		/*------------------------------------ \
			ページ本体 \
		*/ \
		#wrapper { \
			margin-top: initial; \
		} \
		\
		/*==================================== \
			移動元のスペースを詰める \
		*/ \
		/*------------------------------------ \
			ヘッダ \
		*/ \
		.header .layout-wrapper { \
			height: 82px; \
		} \
		/*------------------------------------ \
			サイト名 \
		*/ \
		.header-logo { \
			display: none; \
		} \
		/*------------------------------------ \
			ヘッダ内広告 \
		*/ \
		#header-banner { \
			margin-left: 990px; \
			width: calc(50% - 515px); \
		} \
		#header-banner .multi-ads-area, \
		#header-banner .multi-ads-area > div, \
		#header-banner .multi-ads-area > div > div iframe, \
		#header-banner .multi-ads-area > iframe { \
			width: 100% !important; \
		} \
		\
		/*==================================== \
			BOOTHについて、表示領域の幅が狭い場合 \
		*/ \
		@media screen and (max-width: 1250px) { \
			/*------------------------------------ \
				検索フォーム \
			*/ \
			.global-nav .item-search-box .item-search { \
				position: absolute; \
				top: 48px; \
				left: 0; \
				right: -421px; \
				height: 30px; \
				margin-top: 1px; \
			} \
			\
			/*------------------------------------ \
				検索窓\
			*/ \
			.global-nav .item-search-box .twitter-typeahead { \
				position: absolute !important; \
				width: 356px; \
				top: -40px; \
				left: 150px; \
			} \
			\
			/*------------------------------------ \
				副検索モード \
			*/ \
			.item-search :nth-last-child(-n+2).sub-options label:not(:first-of-type) { \
				right: initial; \
			} \
			.item-search :nth-last-child(-n+2).sub-options::after { \
				right: initial; \
				border-right-color: #FFFFFF; \
			} \
			\
			/*------------------------------------ \
				ページ本体 \
			*/ \
			.page-wrap { \
				padding-top: 78px; \
			} \
			\
			/*------------------------------------ \
				副検索モード \
			*/ \
			.item-search .sub-options::after { \
				height: 6px; \
				top: initial; \
				bottom: 1px; \
			} \
		} \
	</style>');

	/**
	 * 検索オプションを設定するラジオボタンのHTML文字列。
	 * @type {string}
	 */
	var optionsHTML = ' \
		<label> \
			<input data-placeholder="' + h(_('検索')) + '" data-word-name="word" data-form-action="http://www.pixiv.net/search.php" value="s_tag_full" name="s_mode" type="radio" />'
			+ h(_('完全一致')) +
		'</label> \
		<label> \
			<input data-placeholder="' + h(_('検索')) + '" data-word-name="word" data-form-action="http://www.pixiv.net/search.php" value="s_tag" name="s_mode" type="radio" />'
			+ h(_('部分一致')) +
		'</label> \
		<label> \
			<input data-placeholder="' + h(_('検索')) + '" data-word-name="word" data-form-action="http://www.pixiv.net/search.php" value="s_tc" name="s_mode" type="radio" />'
			+ h(_('タイトル・キャプション')) +
		'</label> \
		<div id="s_novel" class="sub-options" title="' + h(_('小説')) + '"> \
			<label> \
				<input data-placeholder="' + h(_('小説検索')) + '" data-word-name="word" data-form-action="http://www.pixiv.net/novel/search.php" value="s_novel" name="s_mode" type="radio" /> \
				<img alt="' + h(_('小説')) + '" src="" /> \
			</label> \
			<label> \
				<input data-mode-name="" data-word-name="tag" data-form-action="/novel/tags.php" value="s_tag_full" name="s_sub_mode" type="radio" /> \
				' + h(_('タグ')) + ' \
			</label> \
			<label> \
				<input value="s_tag" name="s_sub_mode" type="radio" /> \
				' + h(_('キーワード')) + ' \
			</label> \
			<label> \
				<input value="s_tc" name="s_sub_mode" type="radio" /> \
				' + h(_('本文')) + ' \
			</label> \
		</div>' +
		(loggedin ? '<label title="' + h(_('ユーザー')) + '">\
			<input data-placeholder="' + h(_('ユーザー検索')) + '" data-word-name="nick" data-form-action="http://www.pixiv.net/search_user.php" value="s_usr" name="s_mode" type="radio" /> \
				<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 2048 2048" width="16px" height="16px"> \
					<title>' + h(_('ユーザー')) + '</title> \
					<path fill="#809DB8" d="M848,348 l98,-20 q28,0 54,22 q38,32 48,44 q22,26 34,26 q16,0 26,-4 q14,-6 52,2 q32,6 48,24 q18,20 23,42 t5,50 q0,32 2,48 q-4,24 -2,38 q6,36 6,62 t-6,44 t12,28 t38,10 h22 q28,0 40,46 q8,30 8,60 q0,14 -2,30 l-6,46 q-2,16 -6,22 q-6,10 -10,16 t-14,14 l-18,12 l-24,16 q-18,10 -30,18 q-22,62 -50,102 q-34,48 -58,70 l-22,22 q-10,96 -10,118 v6 q0,8 2,16 q0,8 4,22 q4,18 12,28 l20,28 q10,14 30,28 q8,6 46,26 q26,14 62,24 q46,12 82,18 q88,12 138,28 q60,18 102,44 t62,40 t50,46 q28,30 30,32 q20,22 40,98 q20,74 20,140 v66 h-1768 v-72 q0,-42 14,-126 q16,-92 34,-106 q8,-6 32,-34 t46,-48 t62,-46 q36,-24 100,-44 q66,-22 140,-32 q60,-8 102,-22 q40,-12 74,-32 t50,-34 q26,-24 34,-36 q10,-16 16,-30 q12,-18 10,-20 q-4,-6 -2,-6 v-132 q-4,-2 -12,-8 q-8,-4 -28,-22 q-14,-10 -40,-36 q-18,-18 -38,-50 t-32,-64 l-36,-18 q-20,-10 -24,-14 q-4,-2 -12,-18 q-8,-14 -8,-16 q0,-4 -2,-18 t-4,-22 q0,-4 2,-10 t5,-26 t3,-38 q0,-20 8,-38 q10,-32 12,-36 q8,-14 18,-24 t22,-10 h14 q8,-2 18,-12 q12,-12 8,-30 q2,-64 2,-134 q0,-22 8,-50 q10,-38 18,-58 q12,-32 38,-56 t60,-32 l52,-12 h-2 q2,0 6,-2 q10,-2 12,-2 v-6 v-4 q-2,-4 0,-6 q0,-2 8,-4 t22,-2 z" /> \
				</svg> \
			</label>' : '') +
		'<div id="s_group" class="sub-options" title="' + h(_('グループ')) + '"> \
			<label> \
				<input data-placeholder="' + h(_('グループ検索')) + '" data-mode-name="mode" data-word-name="word" data-form-action="http://www.pixiv.net/group/search_group.php" value="s_group" name="s_mode" type="radio" /> \
				<img alt="' + h(_('グループ')) + '" src="" /> \
			</label> \
			<label> \
				<input value="keyword_all" name="s_sub_mode" type="radio" />'
				+ h(_('すべて')) +
			'</label> \
			<label> \
				<input value="tag_full" name="s_sub_mode" type="radio" />'
				+ h(_('完全一致')) +
			'</label> \
			<label> \
				<input value="tag" name="s_sub_mode" type="radio" />'
				+ h(_('部分一致')) +
			'</label> \
			<label> \
				<input value="keyword" name="s_sub_mode" type="radio" />'
				+ h(_('タイトル・キャプション')) +
			'</label> \
		</div> \
		<label title="' + h(_('BOOTH')) + '"> \
			<input data-placeholder="' + h(_('商品検索...')) + '" data-form-action="' + h(_('https://booth.pm/ja/search/')) + '" value="s_booth" name="s_mode" type="radio"> \
			<img src="https://booth.pm/favicon.ico" width="16" alt="' + h(_('BOOTH')) + '" /> \
		</label>';

	var wrapper = document.getElementById('wrapper');
	if (loggedin) {
		// ログイン後、またはBOOTH
		form = word.form;
		submitButton = form.querySelector('[type="submit"]');

		word.required = true;

		if (booth) {
			// 送信ボタンの移動
			var observer;
			var containers = document.getElementsByClassName('twitter-typeahead');
			if (containers[0]) {
				containers[0].appendChild(submitButton);
			} else {
				observer = new MutationObserver(function (mutations, observer) {
					if (mutations.some(function (mutation) {
						return Array.prototype.some.call(mutation.addedNodes, function (node) {
							return node.nodeType === Node.ELEMENT_NODE && node.classList.contains('twitter-typeahead');
						});
					})) {
						observer.disconnect();
						containers[0].appendChild(submitButton);
					}
				});
				observer.observe(form, {
					childList: true,
				});
			}

			// ページが書き換えられたとき
			observer = new MutationObserver(function (mutations, observer) {
				if (mutations.some(function (mutation) { return mutation.target.id === 'mini-carts'; })) {
					var newForm = document.getElementsByClassName('item-search')[0];
					form.dataset.searchParams = newForm.dataset.searchParams;
					newForm.parentElement.replaceChild(form, newForm);
				}
			});
			observer.observe(document, {
				childList: true,
				subtree: true,
			});
		} else {
			// デフォルトの検索オプションを取得・削除
			input = form.s_mode || form.mode;
			subMode = mode = input ? input.value : 's_tag';
			if (input) {
				input.remove();
			}

			// フォームの移動
			wrapper.insertBefore(form, wrapper.firstChild);
		}

		// 検索オプションを設定するラジオボタンの追加
		form.insertAdjacentHTML('beforeend', optionsHTML);
	} else {
		// ログイン前
		// 検索語句の取得
		var searchWord;
		searchParams = new URLSearchParams(window.location.search.replace('?', ''));
		var smallForm = document.getElementsByClassName('search-small')[0];
		if (smallForm) {
			searchWord = smallForm.word.value;
			smallForm.remove();
			// デフォルトの検索オプションを取得
			input = smallForm.s_mode || smallForm.mode;
		} else {
			searchWord = searchParams.get('word') || '';
		}
		subMode = mode = input ? input.value : 's_tag';

		// 検索フォームを構築する
		wrapper.insertAdjacentHTML('afterbegin', '<form class="ui-search"> \
			<div class="container"> \
				<input required="" id="suggest-input" name="word" value="' + h(searchWord) + '" placeholder="' + h(_('検索')) + '"> \
			</div> \
			<input class="submit sprites-search-old" type="submit">'
			+ optionsHTML +
		'</form>');

		form = wrapper.firstElementChild;
		word = form.word;
		submitButton = form.querySelector('[type="submit"]');
	}

	if (booth) {
		mode = 's_booth';
	} else {
		var pathname = window.location.pathname;
		switch (pathname.split('/')[1]) {
			case 'search.php':
				if (!loggedin) {
					mode = subMode = searchParams.get('s_mode');
				}
				break;
			case 'novel':
				mode = 's_novel';
				if (!loggedin) {
					word.placeholder = _('小説検索');
					subMode = searchParams.get('s_mode') || 's_tag';
				}
				if (pathname === '/novel/tags.php') {
					subMode = 's_tag_full';
				}
				break;
			case 'group':
				mode = 's_group';
				if (!loggedin) {
					word.placeholder = _('グループ検索');
					subMode = searchParams.get('mode') || 'keyword_all';
				}
				break;
		}
	}

	form.addEventListener('submit', function (event) {
		var searchMode, subOption, hiddenParams, i, l;

		if (!/\S/.test(word.value)) {
			// 空白以外の文字が入力されていなければ
			word.value = '';
			return;
		}

		searchMode = form.querySelector('[name="s_mode"]:checked');
		if (booth) {
			if (searchMode.value === 's_booth') {
				// BOOTHからBOOTHに移動するとき
				return;
			} else {
				// BOOTHからpixivに移動するとき
				event.stopPropagation();
				event.preventDefault();
				submitButton.removeAttribute('name');
			}
		} else {
			if (searchMode.value === 's_booth') {
				// pixivからBOOTHに移動するとき
				event.preventDefault();
				window.location.assign(searchMode.dataset.formAction + encodeURIComponent(word.value));
				return;
			}
		}

		subOption = form.querySelector('[name="s_sub_mode"]:checked');

		if ((subOption || searchMode).value.endsWith('tag_full') && /[^  ][  ]+[^  ]/.test(word.value)) {
			// 完全一致タグ検索で、先頭・末尾以外に半角スペース・全角スペースが含まれていれば
			if (subOption) {
				searchMode.checked = false;
				searchMode.click();
			} else {
				// イラスト
				searchMode.value = 's_tag';
			}
		}

		form.action = searchMode.dataset.formAction;
		if (searchMode.dataset.wordName) {
			word.name = searchMode.dataset.wordName;
		}
		if (searchMode.dataset.modeName) {
			searchMode.name = searchMode.dataset.modeName;
		} else if (searchMode.dataset.modeName === '') {
			searchMode.removeAttribute('name');
		}

		if (subOption) {
			// 副検索モードが存在すれば
			searchMode.value = subOption.value;
			subOption.removeAttribute('name');
			if (subOption.dataset.formAction) {
				form.action = subOption.dataset.formAction;
			}
			if (subOption.dataset.wordName) {
				word.name = subOption.dataset.wordName;
			}
			if (subOption.dataset.modeName) {
				searchMode.name = subOption.dataset.modeName;
			} else if (subOption.dataset.modeName === '') {
				searchMode.removeAttribute('name');
			}
		}

		if (form.action !== window.location.origin + window.location.pathname) {
			// サービス間をまたぐ場合、不要なパラメーターは送信しない
			Array.prototype.forEach.call(form.querySelectorAll('[type="hidden"]'), function (hiddenParam) {
				hiddenParam.disabled = true;
			});
		}

		if (event.defaultPrevented) {
			// httpsからhttpに移動する際のセキュリティ警告を回避
			var searchParams = new URLSearchParams();
			Array.prototype.forEach.call(form, function (element) {
				if (element.name && !element.disabled && (element.type !== 'checkbox' && element.type !== 'radio' || element.checked)) {
					searchParams.append(element.name, element.value);
				}
			});
			window.location.assign(form.action + '?' + searchParams);
		}
	});

	// 副検索モードの選択
	form.addEventListener('change', function (event) {
		var subOptions = event.target.parentElement.parentElement, subOption;
		if (event.target.name === 's_mode') {
			// 検索モードの選択なら
			if (subOptions.classList.contains('sub-options')) {
				// 副検索モードが存在すれば
				subOptions.querySelector('[value="s_tag"], [value="keyword_all"]').checked = true;
			} else {
				subOption = form.querySelector('[name="s_sub_mode"]:checked');
				if (subOption) {
					subOption.checked = false;
				}
			}
			word.placeholder = event.target.dataset.placeholder;
		} else if (event.target.name === 's_sub_mode') {
			// 副検索モードの選択なら
			subOption = subOptions.firstElementChild.firstElementChild;
			subOption.checked = true;
			word.placeholder = subOption.dataset.placeholder;
		}
	});

	// ラベルをダブルクリックで検索
	form.addEventListener('dblclick', function (event) {
		var target = event.target;
		if (target['matches' in target ? 'matches' : /* Firefox */ 'mozMatchesSelector']('label, label *')) { // Bug 886308 – Implement Element.matches() <https://bugzilla.mozilla.org/show_bug.cgi?id=886308>
			submitButton.click();
		}
	});

	// 現在の検索モードを設定
	form.querySelector(mode === subMode ? '[value="' + mode + '"]' : '#' + mode + ' [value="' + subMode + '"]').click();

	// 副検索モードが右側にはみ出す問題に対処
	if (!booth && document.getElementById('s_group').offsetLeft > 795) {
		document.head.insertAdjacentHTML('beforeend', '<style> \
			#s_group label:not(:first-of-type) { \
				right: -1px; \
			} \
			#s_group::after { \
				right: -1px; \
				border-right-color: transparent; \
			} \
		</style>');
	}
}



/**
 * 挿入された節の親節が、目印となる節の親節か否かを返すコールバック関数。
 * @callback isTargetParent
 * @param {(Document|Element)} parent
 * @returns {boolean}
 */

/**
 * 挿入された節が、目印となる節か否かを返すコールバック関数。
 * @callback isTarget
 * @param {(DocumentType|Element)} target
 * @returns {boolean}
 */

/**
 * 目印となる節が文書に存在するか否かを返すコールバック関数。
 * @callback existsTarget
 * @returns {boolean}
 */

/**
 * 目印となる節が挿入された直後に関数を実行する。
 * @param {Function} main - 実行する関数。
 * @param {isTargetParent} isTargetParent
 * @param {isTarget} isTarget
 * @param {existsTarget} existsTarget
 * @param {Object} [callbacksForFirefox]
 * @param {isTargetParent} [callbacksForFirefox.isTargetParent] - Firefoxにおける{@link isTargetParent}。
 * @param {isTarget} [callbacksForFirefox.isTarget] - Firefoxにおける{@link isTarget}。
 * @param {number} [timeoutSinceStopParsingDocument=0] - DOM構築完了後に監視を続けるミリ秒数。
 * @version 2014-11-25
 * @global
 */
function startScript(main, isTargetParent, isTarget, existsTarget) {
	/**
	 * {@link checkExistingTarget}で{@link startMain}を実行する間隔(ミリ秒)。
	 * @constant {number}
	 */
	var INTERVAL = 10;
	/**
	 * {@link checkExistingTarget}で{@link startMain}を実行する回数。
	 * @constant {number}
	 */
	var LIMIT = 500;

	/**
	 * 実行済みなら真。
	 * @type {boolean}
	 */
	var alreadyCalled = false;

	// 指定した節が既に存在していれば、即実行
	startMain();
	if (alreadyCalled) {
		return;
	}

	// FirefoxのMutationObserverは、HTMLのDOM構築に関して要素をまとめて挿入したと見なすため、isTargetParent、isTargetを変更
	var callbacksForFirefox = arguments[4];
	if (callbacksForFirefox && typeof MozSettingsEvent !== 'undefined') {
		isTargetParent = callbacksForFirefox.isTargetParent || isTargetParent;
		isTarget = callbacksForFirefox.isTarget || isTarget;
	}

	var observer = new MutationObserver(mutationCallback);
	observer.observe(document, {
		childList: true,
		subtree: true,
	});

	var timeoutSinceStopParsingDocument = arguments[5] || 0;
	if (document.readyState === 'complete') {
		// DOMの構築が完了していれば
		onDOMContentLoaded();
	} else {
		document.addEventListener('DOMContentLoaded', onDOMContentLoaded);
	}

	/**
	 * {@link startMain}を実行し、スクリプトが開始されていなければさらに{@link timeoutSinceStopParsingDocument}ミリ秒待機し、
	 * スクリプトが開始されていなければ{@link stopObserving}を実行する。
	 */
	function onDOMContentLoaded() {
		startMain();
		if (timeoutSinceStopParsingDocument === 0) {
			if (!alreadyCalled) {
				stopObserving();
			}
		} else {
			window.setTimeout(function () {
				if (!alreadyCalled) {
					stopObserving();
				}
			}, timeoutSinceStopParsingDocument);
		}
	}

	/**
	 * 目印となる節が挿入されたら、監視を停止し、{@link checkExistingTarget}を実行する。
	 * @param {MutationRecord[]} mutations - A list of MutationRecord objects.
	 * @param {MutationObserver} observer - The constructed MutationObserver object.
	 */
	function mutationCallback(mutations, observer) {
		for (var mutation of mutations) {
			var target = mutation.target;
			if (target.nodeType === Node.ELEMENT_NODE && isTargetParent(target)) {
				// 子が追加された節が要素節で、かつその節についてisTargetParentが真を返せば
				for (var addedNode of mutation.addedNodes) {
					if (addedNode.nodeType === Node.ELEMENT_NODE && isTarget(addedNode)) {
						// 追加された子が要素節で、かつその節についてisTargetが真を返せば
						observer.disconnect();
						checkExistingTarget(0);
						return;
					}
				}
			}
		}
	}

	/**
	 * {@link startMain}を実行し、スクリプトが開始されていなければ再度実行。
	 * @param {number} count - {@link startMain}を実行した回数。
	 */
	function checkExistingTarget(count) {
		startMain();
		if (!alreadyCalled && count < LIMIT) {
			window.setTimeout(checkExistingTarget, INTERVAL, count + 1);
		}
	}

	/**
	 * 指定した節が存在するか確認し、存在すれば{@link stopObserving}を実行しスクリプトを開始。
	 */
	function startMain() {
		if (!alreadyCalled && existsTarget()) {
			stopObserving();
			main();
		}
	}

	/**
	 * 監視を停止する。
	 */
	function stopObserving() {
		alreadyCalled = true;
		if (observer) {
			observer.disconnect();
		}
		document.removeEventListener('DOMContentLoaded', onDOMContentLoaded);
	}
}

/**
 * 国際化・地域化関数の読み込み、ECMAScriptとWHATWG仕様のPolyfill。
 */
function polyfill() {
	/**
	 * DOM関連のメソッド。
	 */
	var DOMUtils = {
		/**
		 * XMLの特殊文字を文字参照に置換する。
		 * @param {string} str - プレーンな文字列。
		 * @returns {string} HTMLとして扱われる文字列。
		 */
		convertSpecialCharactersToCharacterReferences: function (str) {
			return str.replace(/[&<>"']/g, function (specialCharcter) {
				return '&#x' + specialCharcter.charCodeAt(0).toString(16) + ';'
			});
		},
	};
	window.h = DOMUtils.convertSpecialCharactersToCharacterReferences;

	/**
	 * 以下のような形式の翻訳リソース。すべての言語について、msgidは欠けていないものとする。
	 * {@link Gettext.DEFAULT_LOCALE}のリソースを必ず含む。{@link Gettext.ORIGINAL_LOCALE}のリソースは無視される。
	 * {
	 *     'IETF言語タグ': {
	 *         '翻訳前 (msgid)': '翻訳後 (msgstr)',
	 *         ……
	 *     },
	 *     ……
	 * }
	 * @typedef {Object} LocalizedTexts
	 */

	/**
	 * i18n。
	 * @version 2014-07-10
	 */
	window.Gettext = {
		/**
		 * 翻訳対象文字列 (msgid) の言語。IETF言語タグの「language」サブタグ。
		 * @constant {string}
		 */
		ORIGINAL_LOCALE: 'ja',

		/**
		 * クライアントの言語の翻訳リソースが存在しないとき、どの言語に翻訳するか。IETF言語タグの「language」サブタグ。
		 * @constant {string}
		 */
		DEFAULT_LOCALE: 'en',

		/**
		 * 翻訳リソースを追加する。
		 * @param {LocalizedTexts} localizedTexts
		 */
		setLocalizedTexts: function (localizedTexts) {
			this.multilingualLocalizedTexts = localizedTexts;
		},

		/**
		 * クライアントの言語を設定する。
		 * @param {string} clientLang - IETF言語タグ(「language」と「language-REGION」にのみ対応)。
		 */
		setLocale: function (clientLang) {
			var splitedClientLang = clientLang.split('-', 2);
			this.language = splitedClientLang[0].toLowerCase();
			this.langtag = this.language + (splitedClientLang[1] ? '-' + splitedClientLang[1].toUpperCase() : '');
			if (this.language === 'ja') {
				// ja-JPをjaと同一視
				this.langtag = this.language;
			}
		},

		/**
		 * テキストをクライアントの言語に変換する。
		 * @param {string} message - 翻訳前。
		 * @returns {string} 翻訳後。
		 */
		gettext: function (message) {
			// クライアントの言語が翻訳元の言語なら、そのまま返す
			return this.langtag === this.ORIGINAL_LOCALE && message
					// クライアントの言語の翻訳リソースが存在すれば、それを返す
					|| this.langtag in this.multilingualLocalizedTexts && this.multilingualLocalizedTexts[this.langtag][message]
					// 地域下位タグを取り除いた言語タグの翻訳リソースが存在すれば、それを返す
					|| this.language in this.multilingualLocalizedTexts && this.multilingualLocalizedTexts[this.language][message]
					// 既定言語の翻訳リソースが存在すれば、それを返す
					|| this.DEFAULT_LOCALE in this.multilingualLocalizedTexts && this.multilingualLocalizedTexts[this.DEFAULT_LOCALE][message]
					// そのまま返す
					|| message;
		},

		/**
		 * クライアントの言語。{@link Gettext.setLocale}から変更される。
		 * @type {string}
		 * @access private
		 */
		langtag: 'ja',

		/**
		 * クライアントの言語のlanguage部分。{@link Gettext.setLocale}から変更される。
		 * @type {string}
		 * @access private
		 */
		language: 'ja',

		/**
		 * 翻訳リソース。{@link Gettext.setLocalizedTexts}から変更される。
		 * @type {LocalizedTexts}
		 * @access private
		 */
		multilingualLocalizedTexts: {},
	};
	window._ = Gettext.gettext.bind(Gettext);

	// Polyfill for Opera and Google Chrome
	if (typeof URLSearchParams === 'undefined') {
		/** @namespace URL */
		/**
		 * @param {string} input
		 * @returns {Array.<Array.<string>>} List of name-value pairs where both name and value hold a string.
		 * @see [application/x-www-form-urlencoded – URL Standard]{@link https://url.spec.whatwg.org/#concept-urlencoded-string-parser}
		 * @memberof URL
		 * @function
		 * @private
		 */
		var parseXWWWFormUrlencoded = function (input) {
			var output = [];
			for (var bytes of input.split('&')) {
				if (bytes !== '') {
					bytes = bytes.replace(/\+/g, ' ');
					var index = bytes.indexOf('=');
					var name, value;
					if (index !== -1) {
						name = bytes.slice(0, index);
						value = bytes.slice(index + 1);
					} else {
						name = bytes;
						value = '';
					}
					output.push([decodeURIComponent(name), decodeURIComponent(value)]);
				}
			}
			return output;
		};
		/**
		 * @param {string} input - A byte sequence.
		 * @returns {string}
		 * @see [application/x-www-form-urlencoded – URL Standard]{@link https://url.spec.whatwg.org/#concept-urlencoded-byte-serializer}
		 * @memberof URL
		 * @function
		 * @private
		 */
		var serializeXWWWFormUrlencodedByte = function (input) {
			return encodeURIComponent(input).replace(/%20/g, '+').replace(/[!~'()]+/g, escape);
		};
		/**
		 * @param {Array.<Array.string>} pairs - List of name-value pairs pairs.
		 * @returns {string}
		 * @see [application/x-www-form-urlencoded – URL Standard]{@link https://url.spec.whatwg.org/#concept-urlencoded-serializer}
		 * @memberof URL
		 * @function
		 * @private
		 */
		var serializeXWWWFormUrlencodedString = function (pairs) {
			return pairs.map(function (pair) {
				return serializeXWWWFormUrlencodedByte(pair[0]) + '=' + serializeXWWWFormUrlencodedByte(pair[1]);
			}).join('&');
		};

		var _URLSearchParams_list = new WeakMap();
		var _URLSearchParams_urlObjects = new WeakMap();

		/**
		 * A URLSearchParams object has an associated list of name-value pairs, which is initially empty.
		 * @param {(string|URLSearchParams)} [init=""]
		 * @see [URLSearchParams Interface – URL Standard]{@link https://url.spec.whatwg.org/#interface-urlsearchparams}
		 * @version polyfill-2015-07-30
		 * @license CC-BY-4.0
		 * @constructor URLSearchParams
		 */
		Object.defineProperty(window, 'URLSearchParams', {
			writable: true,
			enumerable: false,
			configurable: true,
			value: function (init) {
				if (init === undefined) {
					init = '';
				} else if (!(init instanceof URLSearchParams) && typeof init !== 'string') {
					init = String(init);
				}

				/**
				 * A URLSearchParams object has an associated list of name-value pairs, which is initially empty.
				 * @type {Array.<Array.<string, string>>}
				 * @private
				 * @memberof URLSearchParams#
				 */
				var list = [];
				if (init instanceof URLSearchParams) {
					for (var pair of _URLSearchParams_list.get(init)) {
						list.push(pair[0], pair[1]);
					}
				} else if (init !== '') {
					list = parseXWWWFormUrlencoded(init);
				}
				_URLSearchParams_list.set(this, list);

				/**
				 * A URLSearchParams object has an associated list of zero or more url objects, which is initially empty. 
				 * @type {Set.<Array.<string, string>>}
				 * @private
				 * @member URLSearchParams#urlObjects
				 */
				_URLSearchParams_urlObjects.set(this, new Set());
			}
		});

		/**
		 * @see [URLSearchParams Interface – URL Standard]{@link https://url.spec.whatwg.org/#concept-URLSearchParams-update}
		 * @param {(URL|HTMLAnchorElement|HTMLAreaElement|Location)} [excluded]
		 * @function
		 * @private
		 * @function URLSearchParams#runQueryUpdateSteps
		 */
		var _URLSearchParams_runQueryUpdateSteps = function (excluded) {
			var list = _URLSearchParams_list.get(this);
			var search = excluded
					? excluded.search
					: (list[0] ? '?' + serializeXWWWFormUrlencodedString(list) : '');
			for (var urlObject of _URLSearchParams_urlObjects.get(this)) {
				if (urlObject !== excluded) {
					urlObject.search = search;
				}
			}
		};

		Object.defineProperties(URLSearchParams.prototype, /** @lends URLSearchParams# */ {
			/**
			 * Append a new name-value pair whose name is name and value is value, to the list of name-value pairs.
			 * @param {string} name
			 * @param {string} value
			 * @function
			 */
			append: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function (name, value) {
					if (arguments.length < 2) {
						throw new TypeError('Failed to execute \'append\' on \'URLSearchParams\': 2 argument required, but only ' + arguments.length + ' present.');
					}
					_URLSearchParams_list.get(this).push([String(name), String(value)]);
					_URLSearchParams_runQueryUpdateSteps.call(this);
				}
			},
			/**
			 * Remove all name-value pairs whose name is name.
			 * @param {string} name
			 * @function
			 */
			'delete': {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function (name) {
					if (arguments.length < 1) {
						throw new TypeError('Failed to execute \'delete\' on \'URLSearchParams\': 1 argument required, but only ' + arguments.length + ' present.');
					}
					var list = _URLSearchParams_list.get(this);
					for (var i = 0, l = list.length; i < l; i++) {
						if (list[i][0] === name) {
							list.splice(i, 1);
							i--;
							l--;
						}
					}
					_URLSearchParams_runQueryUpdateSteps.call(this);
				}
			},
			/**
			 * Return the value of the first name-value pair whose name is name, and null if there is no such pair.
			 * @param {string} name
			 * @returns {?string}
			 * @function
			 */
			get: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function (name) {
					if (arguments.length < 1) {
						throw new TypeError('Failed to execute \'get\' on \'URLSearchParams\': 1 argument required, but only ' + arguments.length + ' present.');
					}
					for (var pair of _URLSearchParams_list.get(this)) {
						if (pair[0] === name) {
							return pair[1];
						}
					}
					return null;
				}
			},
			/**
			 * Return the values of all name-value pairs whose name is name, in list order, and the empty sequence otherwise.
			 * @param {string} name
			 * @returns {string[]}
			 * @function
			 */
			getAll: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function (name) {
					if (arguments.length < 1) {
						throw new TypeError('Failed to execute \'getAll\' on \'URLSearchParams\': 1 argument required, but only ' + arguments.length + ' present.');
					}
					var values = [];
					for (var pair of _URLSearchParams_list.get(this)) {
						if (pair[0] === name) {
							values.push(pair[1]);
						}
					}
					return values;
				}
			},
			/**
			 * If there are any name-value pairs whose name is name, set the value of the first such name-value pair to value and remove the others.
			 * Otherwise, append a new name-value pair whose name is name and value is value, to the list of name-value pairs.
			 * @param {string} name
			 * @param {string} value
			 * @function
			 */
			set: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function (name, value) {
					if (arguments.length < 2) {
						throw new TypeError('Failed to execute \'set\' on \'URLSearchParams\': 2 argument required, but only ' + arguments.length + ' present.');
					}
					var list = _URLSearchParams_list.get(this);
					var flag;
					for (var i = 0, l = list.length; i < l; i++) {
						if (list[i][0] === name) {
							if (flag) {
								list.splice(i, 1);
								i--;
								l--;
							} else {
								list[i][1] = String(value);
								flag = true;
							}
						}
					}
					if (!flag) {
						list.push([String(name), String(value)]);
					}
					_URLSearchParams_runQueryUpdateSteps.call(this);
				}
			},
			/**
			 * Return true if there is a name-value pair whose name is name, and false otherwise.
			 * @param {string} name
			 * @returns {boolean}
			 * @function
			 */
			has: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function (name) {
					if (arguments.length < 1) {
						throw new TypeError('Failed to execute \'has\' on \'URLSearchParams\': 1 argument required, but only ' + arguments.length + ' present.');
					}
					for (var pair of _URLSearchParams_list.get(this)) {
						if (pair[0] === name) {
							return true;
						}
					}
					return false;
				}
			},
			/**
			 * Return the serialization of the URLSearchParams object's associated list of name-value pairs.
			 * @returns {string}
			 * @function
			 */
			toString: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function () {
					return serializeXWWWFormUrlencodedString(_URLSearchParams_list.get(this));
				}
			},
			/**
			 * @returns {Iterator.<Array.<string, string>>}
			 * @function
			 */
			entries: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function* () {
					for (var pair of _URLSearchParams_list.get(this)) {
						yield [pair[0], pair[1]];
					}
				}
			},
			/**
			 * @returns {Iterator.<string>}
			 * @function
			 */
			keys: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function* () {
					for (var pair of _URLSearchParams_list.get(this)) {
						yield pair[0];
					}
				}
			},
			/**
			 * @returns {Iterator.<string>}
			 * @function
			 */
			values: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function* () {
					for (var pair of _URLSearchParams_list.get(this)) {
						yield pair[1];
					}
				}
			},
		});
		/**
		 * The value pairs to iterate over are the list name-value pairs with the key being the name and the value the value. 
		 * @returns {Iterator.<Array.<string, string>>}
		 * @function URLSearchParams#@@iterator
		 */
		Object.defineProperty(URLSearchParams.prototype, Symbol.iterator, {
			writable: true,
			enumerable: false,
			configurable: true,
			value: function* () {
				for (var pair of _URLSearchParams_list.get(this)) {
					yield [pair[0], pair[1]];
				}
			}
		});

		/** @namespace URLUtils */
		var _URLUtils_queryObject = new WeakMap();
		/**
		 * @member {URLSearchParams} URLUtilsSearchParams#searchParams
		 */
		for (var interfaceObject of [URL, HTMLAnchorElement, HTMLAreaElement]) {
			Object.defineProperty(interfaceObject.prototype, 'searchParams', {
				enumerable: false,
				configurable: true,
				get: function () {
					var queryObject = _URLUtils_queryObject.get(this);
					if (!queryObject) {
						// If query object is null,
						// set query object to a new URLSearchParams object using query, 
						queryObject = new URLSearchParams(this.search.replace('?', ''));
						_URLUtils_queryObject.set(this, queryObject);
						// and then append the context object to query object’s list of url objects.
						_URLSearchParams_urlObjects.get(queryObject).add(this);
					}
					return queryObject;
				},
				set: function (object) {
					if (!(object instanceof Object)) {
						throw new TypeError('Value being assigned to ' + Object.prototype.toString.call(this).match(/ (.+)\]/)[1] + '.searchParams is not an object.');
					} else if (!(object instanceof URLSearchParams)) {
						throw new TypeError('Value being assigned to ' + Object.prototype.toString.call(this).match(/ (.+)\]/)[1] + '.searchParams does not implement interface URLSearchParams.');
					}
					var queryObject = _URLUtils_queryObject.get(this);
					// Remove the context object from query object’s list of url objects.
					if (queryObject) {
						_URLSearchParams_urlObjects.get(queryObject).delete(this);
					}
					// Append the context object to object’s list of url objects.
					_URLSearchParams_urlObjects.get(object).add(this);
					// Set query object to object.
					_URLUtils_queryObject.set(this, object);
				}
			});
		}
		/**
		 * @private
		 * @function URLUtils#updateSearchParams
		 */
		var _URLUtils_updateSearchParams = function () {
			var searchParams = this.searchParams;
			var search = this.search;
			_URLSearchParams_list.set(searchParams, search ? parseXWWWFormUrlencoded(search.replace('?', '')) : []);
			_URLSearchParams_runQueryUpdateSteps.call(searchParams, this);
		};
		var _pushState = History.prototype.pushState;
		History.prototype.pushState = function () {
			_pushState.apply(this, arguments);
			_URLUtils_updateSearchParams.call(window.location);
		};
		var _replaceState = History.prototype.replaceState;
		History.prototype.replaceState = function () {
			_replaceState.apply(this, arguments);
			_URLUtils_updateSearchParams.call(window.location);
		};
		window.addEventListener('popstate', function (event) {
			_URLUtils_updateSearchParams.call(window.location);
		});
		new MutationObserver(function (mutations) {
			for (var record of mutations) {
				var target = record.target;
				var name = target.localName;
				if (name === 'a' || name === 'area') {
					_URLUtils_updateSearchParams.call(target);
				}
			}
		}).observe(document, {
			attributeFilter: ['href'],
			subtree: true,
		});
	}
}

})();