ニコ生アラート(簡)

Alerts you to live streams that match your search. Supports these sites: FC2 Live, CaveTube, SHOWROOM, Stickam JAPAN!, TwitCasting, Twitch, Niconico Live, Himawari Stream, FRESH! (AbemaTV), Whowatch, Periscope (Twitter)

As of 2019-01-17. See the latest version.

// ==UserScript==
// @name        ニコ生アラート(簡)
// @name:ja     ニコ生アラート (簡)
// @name:en     Nico Live Alert (Kan)
// @description Alerts you to live streams that match your search. Supports these sites:  FC2 Live, CaveTube, SHOWROOM, Stickam JAPAN!, TwitCasting, Twitch, Niconico Live, Himawari Stream, FRESH! (AbemaTV), Whowatch, Periscope (Twitter)
// @description:ja キーワードにヒットしたライブ配信を通知します。次のサイトに対応: FC2ライブ、CaveTube、SHOWROOM、Stickam JAPAN!、ツイキャス、Twitch、ニコニコ生放送、ひまわりストリーム、FRESH! (AbemaTV)、ふわっち、Periscope (Twitter)
// @namespace   http://userscripts.org/users/347021
// @version     5.4.0
// @match       *://*.nicovideo.jp/*
// @match       *://live.fc2.com/*
// @match       https://gae.cavelis.net/*
// @match       https://www.showroom-live.com/*
// @match       *://www.stickam.jp/*
// @match       *://twitcasting.tv/*
// @match       https://www.twitch.tv/*
// @match       *://himast.in/*
// @match       https://freshlive.tv/*
// @match       https://whowatch.tv/*
// @match       https://www.periscope.tv/*
// @match       https://www.youtube.com/*
// @match       https://www.younow.com/*
// @match       https://livestream.com/*
// @require     https://gitcdn.xyz/repo/greasemonkey/gm4-polyfill/a834d46afcc7d6f6297829876423f58bb14a0d97/gm4-polyfill.js
// @require     https://bowercdn.net/c/jsen-0.6.6/dist/jsen.js
// @require     https://cdnjs.cloudflare.com/ajax/libs/favico.js/0.3.10/favico.js
// @require     https://bowercdn.net/c/compare-versions-3.4.0/index.js
// @require     https://greasyfork.org/scripts/17932/code/suppress-prototypejs.js?version=140950
// @require     https://bowercdn.net/c/css-escape-1.5.1/css.escape.js
// @require     https://greasyfork.org/scripts/17895/code/polyfill.js?version=625392
// @require     https://greasyfork.org/scripts/19616/code/utilities.js?version=230651
// @require     https://greasyfork.org/scripts/17896/code/start-script.js?version=112958
// @license     MPL-2.0
// @compatible  Edge 非推奨 / Deprecated
// @compatible  Firefox
// @compatible  Opera
// @compatible  Chrome
// @grant       GM.getValue
// @grant       GM_getValue
// @grant       GM.setValue
// @grant       GM_setValue
// @grant       GM.deleteValue
// @grant       GM_deleteValue
// @grant       GM.registerMenuCommand
// @grant       GM_registerMenuCommand
// @grant       GM.openInTab
// @grant       GM_openInTab
// @grant       GM.xmlHttpRequest
// @grant       GM_xmlhttpRequest
// @connect     live.fc2.com
// @connect     rss.cavelis.net
// @connect     www.showroom-live.com
// @connect     www.stickam.jp
// @connect     twitcasting.tv
// @connect     api.twitch.tv
// @connect     api.search.nicovideo.jp
// @connect     himast.in
// @connect     freshlive.tv
// @connect     api.whowatch.tv
// @connect     api.periscope.tv
// @connect     www.youtube.com
// @noframes
// @run-at      document-start
// @icon        
// @author      100の人
// @homepageURL https://greasyfork.org/scripts/272
// @contributor HADAA
// ==/UserScript==

(function () {
'use strict';

// L10N
Gettext.setLocalizedTexts({
	/*eslint-disable quote-props, max-len */
	'en': {
		//'(取得不可)': '(No data)',
		'検索ワードにヒットしたライブ配信番組': 'Live streams that match your search words',
		'どのライブ配信サービスか': 'Service from',
		'アイコン': 'Icon',
		'プライベート配信か否か': 'Private program or not',
		'限定公開': 'Limited',
		'経過': 'Elapsed',
		'%d 分': '%dm',
		'%d 時間 %u 分': '%dh%um',
		'配信開始からの経過時間': 'Time elapsed since start of live stream',
		'タイトル': 'Title',
		'番組のタイトル': 'Program title',
		'タグ': 'Tags',
		'カテゴリ・タグ': 'Category and tags',
		'配信者': 'Broadcaster',
		'配信者の名前': 'Broadcaster name',
		'説明文': 'Description',
		'来場': 'Visitors',
		'来場者数': 'Number of visitors',
		'%d 人': '%d',
		'コメ数': 'Comments',
		'%d コメ': '%d',
		'総コメント数': 'Total number of comments',
		'コミュニティ': 'Community',
		'コミュニティ・チャンネル': 'Community or channel',
		'%s 更新': 'Last updated %s', // %sは年月日
		//'メンテナンス中': 'Under maintenance',
		//'サーバーダウン': 'Server is down',
		//'オフライン': 'Offline',
		'検索語句': 'Search words',
		'除外するコミュニティ・チャンネルなどの URL': 'Community or channel URLs to be excluded',
		'保存': 'Save',
		'除外 URL リストの取得先を設定': 'Sets the location of the URL exclusion list',
		'特定の URL から、除外 URL のリストを読み込み、検索時に付加します。': 'Loads exclusion list from designated URL and adds to search.',
		'JSON 形式の URL 文字列の配列のみ有効です。': 'Array of URLs needs to be in JSON format.',
		'また、除外 URL リストの読み込みは、アラートページ読み込み時に1回だけ行われます。': 'Also, this script loads exclusion list only once when the alert page is opened.',
		//'GM.xmlHttpRequest エラー': 'GM.xmlHttpRequest error',
		'指定された URL から、除外 URL リストを読み込めませんでした。\n取得せずに続行します。\n\nエラーメッセージ:\n%s': 'Failure to fetch URL exclusion list from designated URL. Continue without fetching.\n\nError message:\n%s',
		'追加設定ボックスの開閉': 'Toggle extra settings',
		'検索対象のライブ配信サービス': 'Live streaming services for search',
		'サービス名': 'Service name',
		'最後に検索結果の取得に成功にした日時': 'Last successful search result timestamp',
		'直近のエラー': 'Last error',
		'FC2ライブ': 'FC2 Live',
		'CaveTube': 'CaveTube',
		'SHOWROOM': 'SHOWROOM',
		'Stickam JAPAN!': 'Stickam JAPAN!',
		'ツイキャス': 'TwitCasting',
		'Twitch': 'Twitch',
		'ニコニコ生放送': 'Niconico Live',
		'ひまわりストリーム': 'Himawari Stream',
		'FRESH! (AbemaTV)': 'FRESH! (AbemaTV)',
		'ふわっち': 'Whowatch',
		'Periscope (Twitter)': 'Periscope (Twitter)',
		'YouTube ライブ': 'YouTube Live',
		'YouNow': 'YouNow',
		'Livestream': 'Livestream',
		'表示する項目の設定': 'Set which items to display',
		'その他の設定': 'Other Settings',
		'プライベート配信を通知しない': 'Do not notify about private programs',
		'タイトル・キャプション・コミュニティ名が %d 文字を超えたら省略する': 'Truncate to %d characters if title description or community name is longer',
		'言語で絞り込む (言語が取得可能なサービスのみ)': 'Filter by language (only for services that have this function)',
		'アラート音': 'Alert sound',
		'ファイルサイズが大きいため、設定に失敗しました。\n\nエラーメッセージ:\n%s': 'Failure to set because file is too large.\n\nError message:\n%s',
		'使用中のブラウザが対応していないファイル形式です。': 'Your browser cannot play this type of file.',
		'項目名クリックで番組を昇順・降順に並べ替えることができます。': 'If you click on item name, you can sort programs.',
		'項目名をドラッグ&ドロップで列の位置を変更できます。': 'Drag and drop item name to change column position.',
		'ユーザー名やコミュニティ名をテキストエリアにドラッグ&ドロップで除外 URL に指定できます。': 'Drag and drop user or community name to textarea to set URL exclusion filter.',
		'RSSの取得に失敗しました。ページを更新してみてください。\n\nエラーメッセージ:\n%s\n%d 行目': 'Failure to read RSS file. Please refresh this page.\n\nError message:\n%s\non line %d',
		'更新しますか?': 'Do you want to refresh?',
		'指定された URL から NG リストを読み込めませんでした。\n取得せずに続行します。\n\nエラーメッセージ:\n%s': 'Failure to get communities and channels from the specified URL.\nScript will continue without getting them.\n\nError message:\n%s',
		'設定のインポートとエクスポート': 'Import and export settings',
		'JSONファイルからインポートする': 'Import from JSON file',
		'JSON形式でファイルにエクスポート': 'Export to file in JSON format',
		'インポートに失敗しました。\n\nエラーメッセージ:\n%s': 'Import failed.\n\nError message:\n%s',
		'インポートが完了しました。ページを再読み込みします。': 'Import completed. Refreshing this page.',
		'ローカルストレージの容量制限を超えたので、プロパティ %p を無視しました。': 'Because the capacity of local storage was exceeded, %p property is ignored.', // %pはカンマ区切りのプロパティ名
		'値が壊れていたので、プロパティ %p を無視しました。': 'Because the value corrupted, %p property is ignored.', // %pはプロパティ名
		'使用中のブラウザが対応していないファイル形式のため、プロパティ %p を無視しました。': 'Because your browser doesn\'t support this type of file, %p property is ignored.', // %pはプロパティ名
		'アラート音を選択': 'Sets alert sound',
		'設定済みのアラート音を削除': 'Deletes alert sound set in advance',
		' ❰❰%s❱❱ ': ' <<%s>> ', // ツールチップ内における一致箇所のマーク

		'ニコ生アラート (簡)': 'Nico Live alert (Kan)',
	},
	/*eslint-enable quote-props, max-len */
});

// クライアントの言語を設定する
Gettext.setLocale(window.navigator.language);

/**
 * HTTPリクエスト。
 * @version 1.0.2
 */
class HTTPRequest
{
	/**
	 * HTTPリクエストに必要な情報。
	 * @typedef {Object} HTTPRequestInit
	 * @see [GM.xmlHttpRequest - GreaseSpot Wiki]{@link https://wiki.greasespot.net/GM.xmlHttpRequest#Arguments}
	 * @see [Fetch Standard (日本語訳)]{@link https://triple-underscore.github.io/Fetch-ja.html#requestinit}
	 * @property {string} method - 「GET」「POST」のいずれか。
	 * @property {string} url
	 * @property {string} responseType - 「document」「json」「text」のいずれか。
	 * @property {Object.<string>} headers
	 * @property {number} [timeout] - ミリ秒。
	 * @property {(string|Object)} [data]
	 * @property {string} mode -
	 *                         {@link XMLHttpRequest} を使用するなら「cors」、{@link GM.xmlHttpRequest} を使用する必要があれば「no-cors」を指定する。
	 */

	/**
	 * @param {HTTPRequestInit} init
	 */
	constructor(init)
	{
		/** @access private */
		this.details = init;
	}

	/**
	 * @access private
	 * @param {Object} client
	 * @param {Function} resolve
	 * @param {Function} reject
	 * @param {string} [responseType]
	 */
	onload(client, resolve, reject, responseType)
	{
		if (client.status === 200) {
			if (client instanceof XMLHttpRequest) {
				if (client.response) {
					resolve(client.response);
				} else {
					reject(new SyntaxError('Parsing HTTP response body was failed.'));
				}
			} else if (client.responseText === '') {
				reject(new SyntaxError('HTTP response body was empty.'));
			} else {
				switch (this.details.responseType) {
					case 'json':
						resolve(JSON.parse(client.responseText));
						break;

					case 'document': {
						let result = /^content-type\s*:\s*(text\/html|text\/xml|application\/xml|\S+\+xml)(?:;|\s|$)/im
							.exec(client.responseHeaders);
						if (result) {
							result = new DOMParser().parseFromString(
								client.responseText,
								result[1] === 'text/html' ? 'text/html' : 'application/xml'
							);

							const parsererror = result.getElementsByTagName('parsererror')[0];
							if (parsererror) {
								reject(new SyntaxError(parsererror.textContent));
							} else {
								if (result.head) {
									result.head.insertAdjacentHTML('beforeend', h`<base href="${client.finalUrl}" />`);
								}
								resolve(result);
							}
						}
						break;
					}
					case 'text':
						resolve(client.responseText);
						break;
				}
			}
		} else {
			console.debug(client.response || client.responseText);
			reject(new ErrorStatusException(client.status));
		}
	}

	/**
	 * @access private
	 * @param {Function} reject
	 */
	ontimeout(reject)
	{
		reject(new TimeoutException());
	}

	/**
	 * @access private
	 * @param {Function} reject
	 */
	onabort(reject)
	{
		reject(new AbortException());
	}

	/**
	 * @access private
	 * @param {Function} reject
	 */
	onerror(reject)
	{
		reject(new NetworkException());
	}

	/**
	 * リクエストを送信します。
	 * @returns {Promise.<string|Object>}
	 */
	send()
	{
		return new Promise((resolve, reject) => {
			if (this.details.mode === 'cors') {
				/**
				 * abort() メソッドを持つオブジェクト。
				 * @type {Object}
				 */
				this.client = new XMLHttpRequest();
				this.client.open(this.details.method, this.details.url);
				this.client.responseType = this.details.responseType;
				this.client.addEventListener('load', event => {
					this.onload(event.target, resolve, reject);
				});
				this.client.addEventListener('timeout', () => {
					this.ontimeout(reject);
				});
				this.client.addEventListener('abort', () => {
					this.onabort(reject);
				});
				this.client.addEventListener('error', () => {
					this.onerror(reject);
				});
				if (this.details.headers) {
					for (const name in this.details.headers) {
						this.client.setRequestHeader(name, this.details.headers[name]);
					}
				}
				if (this.details.timeout) {
					this.client.timeout = this.details.timeout;
				}
				if (this.details.data && typeof this.details.data === 'object') {
					this.client.setRequestHeader('content-type', 'application/json');
					this.client.send(JSON.stringify(this.details.data));
				} else {
					this.client.send(this.details.data);
				}
			} else {
				const details = {};
				for (const key in this.details) {
					details[key] = this.details[key];
				}
				delete details.responseType;
				details.onload = responseObject => {
					this.onload(responseObject, resolve, reject, details.responseType);
				};
				details.ontimeout = () => {
					this.ontimeout(reject);
				};
				details.onabort = () => {
					this.onabort(reject);
				};
				details.onerror = () => {
					this.onerror(reject);
				};
				if (details.data && typeof details.data === 'object') {
					if (!details.headers) {
						details.headers = {};
					}
					details.headers['content-type'] = 'application/json';
					details.data = JSON.stringify(details.data);
				}
				this.client = GM.xmlHttpRequest(details);
			}
		});
	}

	/**
	 * リクエストを取り消します。
	 */
	abort()
	{
		if (this.client) {
			this.client.abort();
		}
	}
}

/**
 * 接続に関する例外。
 */
class ConnectionException extends Error
{
	/**
	 * @param {string} message
	 */
	constructor(message = 'Connection exception occured.')
	{
		super(message);
		this.name = 'ConnectionException';
	}
}

/**
 * 時間切れ。
 */
class TimeoutException extends ConnectionException
{
	/**
	 * @param {string} message
	 */
	constructor(message = 'Connection timed out.')
	{
		super(message);
		this.name = 'TimeoutException';
	}
}

/**
 * HTTPステータスコードが200でない。
 */
class ErrorStatusException extends ConnectionException
{
	/**
	 * @param {number} code - HTTPステータスコード。
	 * @param {string} message
	 */
	constructor(code, message = 'HTTP status-code was %s.'.replace('%s', code))
	{
		super(message);
		this.name = 'ErrorStatusException';

		/**
		 * HTTPステータスコード。
		 * @type {number}
		 */
		this.code = code;
	}
}

/**
 * クライアント側の意図的な切断。
 */
class AbortException extends ConnectionException
{
	/**
	 * @param {string} message
	 */
	constructor(message = 'Request was aborted.')
	{
		super(message);
		this.name = 'AbortException';
	}
}

/**
 * ネットワークエラー。
 */
class NetworkException extends ConnectionException
{
	/**
	 * @param {string} message
	 */
	constructor(message = 'Network error occured.')
	{
		super(message);
		this.name = 'NetworkException';
	}
}

/**
 * 文字列の正規化などを行うユーティリティークラスです。
 */
class Normalizer
{
	/**
	 * 正規化し、連続する空白文字を半角スペースにします。
	 * @param {string} str
	 * @returns {string}
	 */
	static normalize(str)
	{
		return String(str).normalize('NFKC').replace(/\s+/g, ' ');
	}

	/**
	 * 文字種を統一します。
	 * @param {string} normalizedStr - 正規化済みの文字列。
	 * @returns {string}
	 */
	static unifyCases(normalizedStr)
	{
		return StringProcessor.convertToKatakana(normalizedStr).toLocaleLowerCase();
	}
}

/**
 * 文字列とそこに含まれる文字列に関連する処理を行うユーティリティークラスです。
 */
class WordProcessor
{
	/**
	 * OR検索を行います。
	 * @param {string} str
	 * @param {SearchCriteria[]} words
	 * @returns {?SearchCriteria}
	 */
	static orSearch(str, words)
	{
		let searchCriteria = null;
		for (const word of words) {
			if (this.andSearch(str, word)) {
				searchCriteria = word;
				break;
			}
		}
		return searchCriteria;
	}

	/**
	 * AND検索を行います。
	 * @param {string} str
	 * @param {SearchCriteria} word
	 * @returns {boolean}
	 */
	static andSearch(str, word)
	{
		return this.minusSearch(str, word.minus) && word.plus.every(plus => str.includes(plus));
	}

	/**
	 * マイナス検索を行います。
	 * @param {string} str
	 * @param {string[]} minusWord
	 * @returns {boolean} str に minusWord のどの文字列も含まれていなければ真。
	 */
	static minusSearch(str, minusWord)
	{
		return !minusWord.some(minus => str.includes(minus));
	}

	/**
	 * 検索語句が含まれる位置を取得します。
	 * @param {string} data
	 * @param {(SearchCriteria|SearchCriteria[])} searchCriteria
	 * @returns {(number[])[]} [先頭のオフセット, 末尾のオフセット] の配列。重なる部分は一つにまとめます。
	 */
	static getMatches(data, searchCriteria)
	{
		const words = Array.isArray(searchCriteria)
			? searchCriteria.reduce((words, searchCriteria) => words.concat(searchCriteria.plus), [])
			: searchCriteria.plus;

		const unified = Normalizer.unifyCases(data);
		const offsets = [];
		for (const word of words) {
			const index = unified.indexOf(word);
			if (index !== -1) {
				offsets.push([index, index + word.length]);
			}
		}

		// ソート
		offsets.sort((a, b) => a[0] - b[0]);

		// 位置が重なっていたら一つにまとめる
		for (let i = 0, l = offsets.length; i < l; i++) {
			if (offsets[i + 1] && offsets[i][1] >= offsets[i + 1][0]) {
				// 次の位置のペアが存在し、現在のペアの終了位置が次のペアの開始位置以上であれば
				// 現在のペアの終了位置を次のペアの終了位置に
				offsets[i][1] = offsets[i + 1][1];
				// 次のペアを削除
				offsets.splice(i + 1, 1);
				// 次の次のペアと重なっているかも確認
				i--;
			}
		}

		return offsets;
	}

	/**
	 * 指定された箇所の範囲を作成します。
	 * @param {Text} text
	 * @param {(number[])[]} offsets - 重なる部分が無い [先頭のオフセット, 末尾のオフセット] の配列。
	 * @returns {(Range[]|Text)}
	 */
	static convertOffsetsToRanges(text, offsets)
	{
		return offsets.map(function (offset) {
			const range = new Range();
			range.setStart(text, offset[0]);
			range.setEnd(text, offset[1]);
			return range;
		});
	}

	/**
	 * 指定された部分を括弧で囲みます。
	 * @param {string} data
	 * @param {(number[])[]} offsets - 昇順に並んでいる、重なる部分が無い [先頭のオフセット, 末尾のオフセット] の配列。
	 * @returns {string}
	 */
	static markMatchesAsPlainText(data, offsets)
	{
		return offsets.concat().reverse().reduce(function (data, offset) {
			return data.slice(0, offset[0])
				+ _(' ❰❰%s❱❱ ').replace('%s', data.slice(offset[0], offset[1]))
				+ data.slice(offset[1]);
		}, data);
	}

	/**
	 * 指定された範囲が含まれるようにTextノードを切り出します。
	 * @param {Text} text
	 * @param {(number[])[]} offsets - 昇順に並んでいる、重なる部分が無い [先頭のオフセット, 末尾のオフセット] の配列。
	 * @param {number} maxLength - 表示する最大の符号単位数。切り取ったときにサロゲートペアが壊れるようであれば、1、2文字増やします。
	 * @param {number} beforeLength - 一致箇所より前に表示する符号単位数。切り取ったときにサロゲートペアが壊れるようであれば、1文字増やします。
	 * @returns {string[]} 先頭が切り取られていれば「ellipsis-left」、末尾が切り取られていれば「ellipsis-right」を含む配列。
	 */
	static extractTextNode(text, offsets, maxLength, beforeLength)
	{
		/**
		 * 切り取り範囲。
		 * @type {Range[]}
		 */
		const trimRanges = [];

		/**
		 * 切り取る前の文字列。
		 * @type {number}
		 */
		const data = text.data;

		/**
		 * 切り取る前の文字列の符号単位数。
		 * @type {number}
		 */
		const dataLength = text.length;

		/**
		 * 戻り値。
		 * @type {string[]}
		 */
		const classes = [];

		/**
		 * 表示する部分の終了位置。
		 * @type {number}
		 */
		let viewEndOffset;

		if (offsets.length > 0) {
			// 検索語句が一致する箇所があれば
			/**
			 * 表示する部分の開始位置。
			 * @type {number}
			 */
			let viewStartOffset;

			if (offsets[offsets.length - 1][1] <= maxLength) {
				// 先頭から制限文字数の範囲内にマーク位置がすべて含まれていれば
				viewStartOffset = 0;
				viewEndOffset = maxLength;
			} else {
				viewStartOffset = offsets[0][0] - beforeLength;
				viewEndOffset = viewStartOffset + maxLength;
				if (viewStartOffset < 0) {
					// 表示する部分の開始位置が先頭を超えていれば
					viewStartOffset = 0;
				}
				if (viewEndOffset >= dataLength) {
					// 表示する部分の終了位置が末尾を超えていれば
					// 終了位置を末尾に
					viewEndOffset = dataLength;
					// 開始位置を終了位置から最大文字数分引いた位置に
					viewStartOffset = viewEndOffset - maxLength;
				}
			}

			if (viewStartOffset > 0) {
				// 表示部分の開始位置が先頭より後ろなら
				const charCode = data.charCodeAt(viewStartOffset);
				if (0xDC00 <= charCode && charCode <= 0xDFFF) {
					// 表示部分の先頭文字が下位サロゲートであれば
					viewStartOffset--;
				}
				if (viewStartOffset > 0) {
					const range = new Range();
					range.setStart(text, 0);
					range.setEnd(text, viewStartOffset);
					trimRanges.push(range);
					classes.push('ellipsis-left');
				}
			}
		} else {
			viewEndOffset = maxLength;
		}

		if (viewEndOffset < dataLength) {
			// 表示部分の終了位置が末尾より前なら
			const charCode = data.charCodeAt(viewEndOffset - 1);
			if (0xD800 <= charCode && charCode <= 0xDBFF) {
				// 表示部分の末尾文字が上位サロゲートであれば
				viewEndOffset++;
			}
			if (viewEndOffset < dataLength) {
				const range = new Range();
				range.setStart(text, viewEndOffset);
				range.setEnd(text, dataLength);
				trimRanges.push(range);
				classes.push('ellipsis-right');
			}
		}

		// 切り取る
		for (const range of trimRanges) {
			range.deleteContents();
		}

		return classes;
	}
}



/**
 * メインの処理を行うユーティリティークラス。
 */
class Alert
{
	/**
	 * ページタイトル、ブラウジングコンテキスト名、ユーザースクリプトのコマンド名に用いる文字列。
	 * @constant {string}
	 */
	static get NAME() {return _('ニコ生アラート (簡)');}

	/**
	 * {@link UserSettings.setLargeString} や URL に用いる文字列。
	 * @constant {string}
	 */
	static get ID() {return 'alert-keyword-347021';}

	/**
	 * ページのFaviconに用いるデータURL。
	 * @constant {string}
	 */
	static get ICON() {return '';}

	/**
	 * 初期化します。
	 */
	static initialize()
	{
		// 各サービスのインスタンスにイベントリスナーを設定
		for (const service of this.services) {
			service.addEventListener('progress', this.onprogress);
			service.addEventListener('load', this.onload.bind(this));
			service.addEventListener('error', this.onerror);
		}

		// Faviconの設定
		this.favico = new Favico({
			animation: 'none',
			position : 'up',
		});
	}

	/**
	 * @access private
	 * @param {Service#ProgramEvent} event
	 */
	static onprogress(event)
	{
		const program = event.detail;
		if (!program.private || !document.getElementsByName('exclusionMemberOnly')[0].checked) {
			// プライベート配信で無い、またはプライベート配信を拒否していなければ
			const urls = [program.link];
			if (program.author && program.author.url) {
				urls.push(program.author.url);
			}
			if (program.community && program.community.url) {
				urls.push(program.community.url);
			}
			if (urls.every(url => UserSettings.exclusions.indexOf(url) === -1)) {
				// 除外リストに含まれていなければ
				// 一覧に追加
				TableProcessor.insertProgram(program, urls);

				// アラート音を鳴らす
				const alertTone = document.getElementById('alert-tone');
				if (!alertTone.hidden && !alertTone.muted && alertTone.volume > 0) {
					alertTone.play();
				}
			}
		}
	}

	/**
	 * @access private
	 * @param {Service#LoadedEvent} event
	 */
	static onload(event)
	{
		// 配信終了の番組を削除
		TableProcessor.removeOldPrograms(event.detail.service, event.detail.programs, event.detail.searchCriteria);

		// 最終更新日時を設定
		UserSettings.showLatestUpdatedDate(event.target);

		// ヒット数の更新
		this.showHits();
	}

	/**
	 * ヒット数を更新します。
	 */
	static showHits()
	{
		const tBody = document.querySelector('#programs tbody');

		/**
		 * ヒット数。
		 * @type {number}
		 */
		const hits = tBody.rows.length;

		// ページタイトルの修正
		document.title = (hits > 0 ? `(${hits})` : '') + Alert.NAME;

		// Faviconの修正
		this.favico.badge(hits);

		// 行の色分けの調整
		if (hits % 2 === 0) {
			tBody.classList.remove('odd');
		} else {
			tBody.classList.add('odd');
		}
	}

	/**
	 * スクリプトが停止している状態であれば起動し、動作している状態であればOR検索ができないサービスを再読み込みします。
	 * 検索語句が空の場合は、スクリプトを停止します。
	 */
	static restart()
	{
		if (UserSettings.words.length === 0) {
			// 検索語句が空であれば
			this.stop();
		} else if (this.run) {
			// スクリプトが動作している状態であれば
			for (const service of this.services) {
				if (service.disabledOr) {
					service.reset();
				}
			}
		} else {
			// スクリプトが停止している状態であれば
			// 有効なサービスで検索を開始
			/**
			 * スクリプトが動作している状態であれば真。
			 * @access private
			 * @type {boolean}
			 */
			this.run = true;
			const enabledServices = UserSettings.getTargetServices();
			for (const service of this.services) {
				if (enabledServices.indexOf(service.id) !== -1) {
					service.start();
				}
			}
		}
	}

	/**
	 * スクリプトを停止します。
	 */
	static stop()
	{
		if (this.run) {
			// スクリプトが動作している状態であれば、すべてのサービスで検索を停止
			this.run = false;
			for (const service of this.services) {
				service.stop();
			}
		}
	}

	/**
	 * 指定されたサービスを有効化します。
	 * @param {string} id
	 */
	static enableService(id)
	{
		if (this.run) {
			this.services.find(service => service.id === id).start();
		}
	}

	/**
	 * 指定されたサービスを無効化します。
	 * @param {string} id
	 */
	static disableService(id)
	{
		if (this.run) {
			const service = this.services.find(service => service.id === id);
			service.stop();
			TableProcessor.removeProgramsWithService(service);
		}
	}
}

/**
 * AND検索を行う検索語句。
 * @typedef {Object} SearchCriteria
 * @property {string} plus - AND検索を行うキーワード。
 * @property {string} minus - マイナス検索を行うキーワード。
 */

/**
 * 検索語句にヒットする番組が見つかったとき。
 * @event Service#ProgramEvent
 * @type {CustomEvent}
 * @property {string} type - 「progress」を返す。
 * @property {Program} detail - ライブ配信番組。
 */

/**
 * 最後のページを取得し終えたとき。OR検索できないサービスの場合は、一度の検索で複数回送出される。
 * @event Service#LoadedEvent:load
 * @type {CustomEvent}
 * @property {string} type - 「load」を返す。
 * @property {Object} detail
 * @property {Service} detail.service - 対象のサービス。
 * @property {Program[]} detail.programs - 今回の検索でヒットしたライブ配信番組。OR検索できないサービスの場合は、一つの検索条件にヒットした番組のみ。
 * @property {SearchCriteria} [detail.searchCriteria] - OR検索ができないサービスにおいて、対象の検索条件。
 */

/**
 * 配信の取得に必要な情報を返す。
 * @callback getDetails
 * @param {(SearchCriteria|SearchCriteria[])} [words] - OR検索が可能なら配列となる。
 * @return {HTTPRequestInit}
 */

/**
 * 取得した情報から番組を取り出す。
 * @callback parseResponse
 * @param {(Object|string)} response
 * @param {HTTPRequestInit} details
 * @param {SearchCriteria[]} [words]
 * @return {(Programs|Error)} 取得に失敗している場合は例外を返す。
 */

/**
 * 取得した番組の一覧。
 * @typedef {Object} Programs
 * @property {(Object[]|NodeList|HTMLCollection)} programs - サイト独自形式の番組情報の配列。
 * @property {HTTPRequestInit} [next] - 結果が複数ページにわたる場合に、次のページが存在すれば指定。
 */

/**
 * サイト独自形式の番組情報を {@link Program} に変換する関数。
 * @callback convertIntoEntry
 * @property {Object} programs - サイト独自形式の番組情報。
 * @returns {Program}
 */

/**
 * ライブ配信サービス。
 * @augments EventTarget
 */
class Service
{
	/**
	 * タイムアウトミリ秒数。
	 * @constant {number}
	 */
	static get TIMEOUT() {return 10 * DateUtils.SECONDS_TO_MILISECONDS;}

	/**
	 * メンテナンス中など、サーバー側のエラーが発生した場合に再度取得する間隔。ミリ秒。
	 * @constant {number}
	 */
	static get RETRY_DELAY() {return 15 * DateUtils.MINUTES_TO_MILISECONDS;}

	/**
	 * @param {Object} details
	 * @param {string} details.id - サービスを識別するID。
	 * @param {string} details.name - サイト名。
	 * @param {string} details.url - サイトのURL。
	 * @param {string} [details.icon] - サイトアイコンのURL。サイトのURLがホストとスラッシュで終わり、アイコンが /favicon.ico に配置されている場合は省略可。
	 * @param {getDetails} details.getDetails - 番組の取得に必要な情報を返す。
	 * @param {parseResponse} details.parseResponse - 取得した情報から番組を取り出す。
	 * @param {boolean} [details.disabledSearch] - 全件取得が可能な (検索ができない) サービスなら真。
	 * @param {boolean} [details.disabledOr] - OR検索ができないサービスなら真。
	 * @param {boolean} [details.disabledMinus] - マイナス検索ができないサービスなら真。
	 * @param {boolean} [details.disabledLanguageFilter] - 言語の絞り込み検索ができないサービスなら真。
	 * @param {convertIntoEntry} details.convertIntoEntry - 第1引数のサイト独自形式の番組情報を {@link Program} に変換する関数。
	 * @param {number} [details.delay=360000] - 情報を取得する間隔。ミリ秒。OR検索できないサービスの場合、各単語の検索間隔。既定値は6分。
	 * @param {number} [details.pagingDelay=10000] - 結果が複数ページにわたる場合に、次のページを取得するまでの間隔。ミリ秒。既定値は10秒。
	 */
	constructor(details)
	{
		/**
		 * サービスを識別するID。
		 * @type {string}
		 * @readonly
		 */
		this.id = details.id;

		/**
		 * サイト名。
		 * @type {string}
		 * @readonly
		 */
		this.name = details.name;

		/**
		 * サイトのURL。
		 * @type {string}
		 * @readonly
		 */
		this.url = details.url;

		/**
		 * サイトアイコンのURL。
		 * @type {string}
		 * @readonly
		 */
		this.icon = details.icon || details.url + 'favicon.ico';

		/**
		 * OR検索ができないサービスについて、検索語句の0から始まるインデックス。
		 * @type {number}
		 * @access private
		 */
		this.wordIndex = 0;

		/**
		 * 検索対象のサービスなら真。
		 * @type {boolean}
		 * @access private
		 */
		this.enabled = false;

		/**
		 * abort() メソッドを持つオブジェクト。
		 * @type {HTTPRequest}
		 * @access private
		 */
		this.request;

		/**
		 * タイマーID。
		 * @type {number}
		 * @access private
		 */
		this.timer;

		/**
		 * 現在の検索で取得した番組。
		 * @type {Program[]}
		 * @access private
		 */
		this.currentPrograms = [];

		/**
		 * OR検索できないサービスなら真。
		 * @type {boolean}
		 */
		this.disabledOr = details.disabledOr;

		/**
		 * {@link Service#getHitPrograms}で使用する情報。
		 * @access private
		 * @type {Object}
		 */
		this.details = details;

		// EventTargetの疑似継承
		// <https://stackoverflow.com/a/24216547>
		const eventTarget = new DocumentFragment();
		for (const key in this) {
			eventTarget[key] = typeof this[key] === 'function' ? this[key].bind(this) : this[key];
		}
		for (const methodName of ['addEventListener', 'removeEventListener', 'dispatchEvent']) {
			this[methodName] = eventTarget[methodName].bind(eventTarget);
		}
	}

	/**
	 * 検索を開始します。
	 */
	start()
	{
		if (!this.enabled) {
			// 検索が無効であれば
			this.enabled = true;
			this.getHitPrograms();
		}
	}

	/**
	 * 検索を停止します。
	 */
	stop()
	{
		this.enabled = false;
		if (this.request) {
			this.request.abort();
		}
		window.clearTimeout(this.timer);
	}

	/**
	 * 検索語句を読み込み直して最初から検索しなおします。
	 */
	reset()
	{
		if (this.disabledOr) {
			this.wordIndex = 0;
			TableProcessor.removeProgramsWithService(this);
		}
	}

	/**
	 * 情報を取得する間隔の既定値。ミリ秒。
	 * @access private
	 * @constant {number}
	 */
	static get DEFAULT_DELAY() {return 6 * DateUtils.MINUTES_TO_MILISECONDS;}

	/**
	 * 結果が複数ページにわたる場合に、次のページを取得するまでの間隔の既定値。ミリ秒。
	 * @access private
	 * @constant {number}
	 */
	static get DEFAULT_PAGING_DELAY() {return 1 * DateUtils.SECONDS_TO_MILISECONDS;}

	/**
	 * 検索ワードにヒットした番組を繰り返し取得します。
	 * @access private
	 * @param {HTTPRequestInit} [nextSearchInit] - 次のページの取得に必要な情報。
	 * @fires Service#ProgramEvent:progress
	 * @fires Service#LoadedEvent:load
	 */
	getHitPrograms(nextSearchInit)
	{
		if (this.enabled) {
			/**
			 * 現在の検索ワード。接続前と取得完了時の検索ワードの変化を防ぎます。
			 * @type {SearchCriteria[]}
			 */
			const words = UserSettings.words;

			const searchInit = nextSearchInit
				|| this.details.getDetails(this.details.disabledOr ? words[this.wordIndex] : words);
			searchInit.timeout = Service.TIMEOUT;
			this.request = new HTTPRequest(searchInit);

			this.request.send().then(response => {
				const programs = this.details.parseResponse(response, searchInit, words);
				if (programs instanceof Error) {
					return Promise.reject(programs);
				}

				// 番組情報の取得に成功していれば
				for (const program of Array.from(programs.programs)) {
					const entry = this.details.convertIntoEntry(program);
					entry.service = this;
					if (this.details.disabledOr) {
						// OR検索ができないサービスなら
						entry.searchCriteria = words[this.wordIndex];
					}

					// 検索条件に一致する番組か確認
					if (entry.searchCriteria) {
						// OR検索ができないサービス、またはOR検索が可能でヒットした番組に対応する検索条件を取得可能なサービスなら
						if (!WordProcessor.andSearch(entry.getSearchTarget(), entry.searchCriteria)) {
							delete entry.searchCriteria;
						}
					} else {
						// 全件取得が可能なサービス、またはOR検索が可能なサービスなら
						entry.searchCriteria = WordProcessor.orSearch(entry.getSearchTarget(), words);
					}

					if (entry.searchCriteria && !(document.getElementsByName('languageFilter')[0].checked
						&& this.details.disabledLanguageFilter
						&& window.navigator.language.split('-')[0] !== entry.language.split('-')[0])) {
						// 検索条件に一致する番組、かつ言語が一致する番組なら
						entry.service = this;
						this.currentPrograms.push(entry);
						this.dispatchEvent(new CustomEvent('progress', { detail: entry }));
					}
				}

				if (programs.next) {
					// 次のページが存在すれば
					this.timer = window.setTimeout(
						this.getHitPrograms.bind(this),
						this.details.pagingDelay || Service.DEFAULT_PAGING_DELAY,
						programs.next
					);
				} else {
					// 取得完了
					this.dispatchEvent(new CustomEvent('load', { detail: {
						service: this,
						programs: this.currentPrograms,
						searchCriteria: this.details.disabledOr ? words[this.wordIndex] : null,
					}}));
					this.currentPrograms = [];

					if (this.details.disabledOr) {
						// OR検索ができないサービスなら
						this.wordIndex++;
						if (!UserSettings.words[this.wordIndex]) {
							// 次の検索語句が存在しなければ
							this.wordIndex = 0;
						}
					}

					this.timer = window.setTimeout(
						this.getHitPrograms.bind(this),
						this.details.delay || Service.DEFAULT_DELAY
					);
				}
			}).catch(error => {
				// 番組情報の取得に失敗していれば
				if (!(error instanceof AbortException)) {
					// 意図的な停止による例外でなければ
					console.error(error);
					UserSettings.showLatestError(this, error);
					// 一定時間後に最初から検索をやり直す
					this.timer = window.setTimeout(
						this.getHitPrograms.bind(this),
						error instanceof NetworkException
							? this.details.delay || Service.DEFAULT_DELAY
							: Service.RETRY_DELAY
					);
				}
			});
		}
	}
}

/**
 * 配信者、またはコミュニティ。
 * @typedef {Object} Person
 * @property {string} name - 名前。
 * @property {string} [url] - URL。
 */

/**
 * 1つのライブ配信番組。
 */
class Program
{
	/**
	 * @param {string} link - 配信のURL。
	 * @param {string} title - 配信のタイトル。
	 * @param {Object} otherDetails
	 * @param {string} [otherDetails.icon] - コミュニティやチャンネルなどのアイコンのURL。取得できなければユーザーのアイコンのURL。それも取得できなければ配信のアイコンのURL。
	 * @param {boolean} [otherDetails.private] - プライベート配信であれば真。
	 * @param {Date} [otherDetails.published] - 配信開始日時。
	 * @param {Person} [otherDetails.author] - 配信者。
	 * @param {string[]} [otherDetails.categories] - 配信のタグ。カテゴリを含みます。
	 * @param {string} [otherDetails.summary] - 配信の説明文。
	 * @param {number} [otherDetails.visitors] - 累計視聴者数。取得できなければ最高同時視聴者数。それも取得できなければ現在の視聴者数。
	 * @param {number} [otherDetails.comments] - コメントの数。
	 * @param {Person} [otherDetails.community] - コミュニティやチャンネルなど。
	 * @param {string} [otherDetails.language] - 言語。
	 * @param {SearchCriteria} [searchCriteria] - 検索条件。
	 */
	constructor(link, title, otherDetails, searchCriteria)
	{
		this.link = link;
		this.title = Normalizer.normalize(title);
		for (const key in otherDetails) {
			let value = otherDetails[key];
			if (value !== undefined && value !== null) {
				switch (key) {
					case 'categories':
						value = value.map(Normalizer.normalize);
						break;
					case 'summary':
						value = Normalizer.normalize(value);
						break;
					case 'community':
						value.name = Normalizer.normalize(value.name);
						break;
				}
				this[key] = value;
			}
		}
		this.searchCriteria = searchCriteria;

		/**
		 * @type {Service}
		 */
		this.service;
	}

	/**
	 * 検索対象を取得します。
	 * @returns {string} 文字種を統一した文字列。
	 */
	getSearchTarget()
	{
		let target = [this.title];
		if (this.categories) {
			target = target.concat(this.categories);
		}
		if (this.summary) {
			target.push(this.summary);
		}
		if (this.community) {
			target.push(this.community.name);
		}
		return Normalizer.unifyCases(target.join(' '));
	}
}

/**
 * ライブ配信番組を表示する表に関する処理を行うユーティリティークラス。
 */
const TableProcessor = {
	/**
	 * 経過時間を更新する間隔。ミリ秒数。
	 * @type {number}
	 */
	DURATION_UPDATING_INTERVAL: 20 * DateUtils.SECONDS_TO_MILISECONDS,

	/**
	 * 現在時刻から指定した時刻を引いた差を返します。
	 * @param {Date} value
	 * @returns {Object.<string>} dateTimeプロパティに ISO 8601 形式の文字列 (負になる場合は「PT0S」)、textプロパティに「○時間○分」のような形式の文字列。
	 */
	getDuration: function (value) {
		const milliseconds = Date.now() - value.getTime();
		let minutes = Math.round(milliseconds / DateUtils.MINUTES_TO_MILISECONDS);
		const sign = minutes >= 0 ? 1 : -1;
		minutes = Math.abs(minutes);
		const hours = Math.floor(minutes / 60);
		minutes = minutes % 60;
		return {
			dateTime: `PT${sign === -1 ? 0 : milliseconds / DateUtils.SECONDS_TO_MILISECONDS}S`,
			text: hours
				? _('%d 時間 %u 分').replace('%d', sign * hours).replace('%u', minutes)
				: _('%d 分').replace('%d', sign * minutes),
		};
	},

	/**
	 * 経過時間の更新を開始します。
	 */
	startUpdatingDurations: function () {
		for (const duration of document.querySelectorAll(
			'[itemtype="http://schema.org/VideoObject"] [itemprop="duration"]:not([hidden])'
		)) {
			const serialized = this.getDuration(new Date(duration.dataset.start));
			duration.dateTime = serialized.dateTime;
			duration.textContent = serialized.text;
		}
		window.setTimeout(this.startUpdatingDurations.bind(this), this.DURATION_UPDATING_INTERVAL);
	},

	/**
	 * プライベート番組を表から取り除きます。
	 */
	removePrivatePrograms: function () {
		for (const requiresSubscription of document.querySelectorAll(
			'[itemtype="http://schema.org/VideoObject"] [itemprop="requiresSubscription"][value="true"]'
		)) {
			requiresSubscription.closest('[itemscope]').remove();
		}
		Alert.showHits();
	},

	/**
	 * 指定されたサービスの番組を表から取り除きます。
	 * @param {Service} service
	 */
	removeProgramsWithService: function (service) {
		for (const row of this.getProgramsWithService(service)) {
			row.remove();
		}
		Alert.showHits();
	},

	/**
	 * 指定されたサービスの番組を取得します。
	 * @param {Service} service
	 * @param {SearchCriteria} [searchCriteria] - OR検索ができないサービスにおいて、対象の検索条件。
	 * @returns {HTMLRowElement[]}
	 */
	getProgramsWithService: function (service, searchCriteria) {
		return Array.from(document.querySelectorAll(
			'[itemtype="http://schema.org/VideoObject"]'
				+ ' [itemprop="provider"]'
					+ (searchCriteria ? `[data-search-criteria="${CSS.escape(JSON.stringify(searchCriteria))}"]` : '')
				+ ` [itemprop="url"][href="${service.url}"]`
		)).map(providerURL => providerURL.closest('[itemtype="http://schema.org/VideoObject"]'));
	},

	/**
	 * 除外対象の番組を表から取り除きます。
	 */
	removeExclusions: function () {
		for (const urlProperty of document.querySelectorAll(
			'[itemtype="http://schema.org/VideoObject"] [itemprop="url"],'
				+ '[itemtype="http://schema.org/VideoObject"] [itemprop="workLocation"]'
		)) {
			if (UserSettings.exclusions.indexOf(urlProperty.href) !== -1) {
				const row = urlProperty.closest('[itemtype="http://schema.org/VideoObject"]');
				if (row.parentElement) {
					row.remove();
				}
			}
		}
		Alert.showHits();
	},

	/**
	 * 以前に取得した番組を表から取り除きます。
	 * @param {Service} service - 対象のサービス。
	 * @param {Program[]} currentPrograms - 今回取得した番組。
	 * @param {SearchCriteria} [searchCriteria] - OR検索ができないサービスにおいて、対象の検索条件。
	 */
	removeOldPrograms: function (service, currentPrograms, searchCriteria) {
		const urls = currentPrograms.map(program => program.link);
		for (const row of this.getProgramsWithService(service, searchCriteria)) {
			if (urls.indexOf(row.querySelector('td > [itemprop="url"]').href) === -1) {
				row.remove();
			}
		}
	},

	/**
	 * 指定された番組を表に追加します。
	 * @param {Program} program
	 * @param {string[]} urls - 番組、ユーザー、コミュニティのURL。
	 */
	insertProgram: function (program, urls) {
		const table = document.getElementById('programs');
		const anchor = table.querySelector(urls.map(url => `[href="${CSS.escape(program.link)}"]`).join(','));
		let previousRow;
		if (anchor) {
			// すでに同じ番組、または同じユーザーの番組、同じコミュニティの番組が追加されていれば
			previousRow = anchor.closest('[itemtype="http://schema.org/VideoObject"]');
		}

		const row = this.convertProgramToRow(program, previousRow);
		const tBody = table.tBodies[0];
		tBody.insertBefore(row, tBody.rows[this.getInsertPosition(table, row)]);
	},

	/**
	 * 指定された番組を表すtr要素を返します。
	 * @param {Program} program
	 * @param {HTMLTableRowElement} [previousRow] - 指定されていれば、その行を更新します。
	 * @returns {HTMLTableRowElement}
	 */
	convertProgramToRow: function (program, previousRow) {
		const row
			= previousRow || document.querySelector('#programs template').content.firstElementChild.cloneNode(true);

		// サービス
		if (!previousRow) {
			const provider = row.querySelector('[itemprop="provider"]');
			provider.querySelector('[itemprop="name"]').value = program.service.name;
			provider.querySelector('[itemprop="url"]').href = program.service.url;
			const logo = provider.querySelector('[itemprop="logo"]');
			logo.src = program.service.icon;
			logo.alt = program.service.name;
			logo.title = program.service.name;
			if (program.service.disabledOr) {
				provider.dataset.searchCriteria = JSON.stringify(program.searchCriteria);
			}
		}

		// アイコン
		const image = row.querySelector('[itemprop="image"]');
		if (program.icon) {
			if (!previousRow || program.icon !== image.src) {
				image.src = program.icon;
				image.hidden = false;
			}
		} else {
			image.hidden = true;
		}

		// コミュ限
		const requiresSubscription = row.querySelector('[itemprop="requiresSubscription"]');
		if (program.private) {
			if (!previousRow || requiresSubscription.value === 'false') {
				requiresSubscription.value = 'true';
				requiresSubscription.textContent = _('限定公開');
			}
		} else {
			if (previousRow && requiresSubscription.value === 'true') {
				requiresSubscription.value = 'false';
				requiresSubscription.textContent = '';
			}
		}

		// 経過時間
		const duration = row.querySelector('[itemprop="duration"]');
		if (program.published) {
			if (!previousRow || program.published.toISOString() !== duration.dataset.start) {
				const serialized = this.getDuration(program.published);
				duration.dateTime = serialized.dateTime;
				duration.textContent = serialized.text;
				duration.dataset.start = program.published.toISOString();
				duration.hidden = false;
			}
		}

		// タイトル
		const name = row.querySelector('td:not([itemprop="provider"]) > [itemprop="name"]');
		if (!previousRow || program.title !== name.value) {
			name.value = program.title;
			this.setMarkedText(name.firstElementChild, program.title, program.searchCriteria || UserSettings.words);
		}

		// タグ
		const keywords = row.querySelector('[itemprop="keywords"]');
		const tags = program.categories ? program.categories.join(',') : '';
		if (tags !== keywords.value) {
			keywords.value = tags;
			this.setMarkedText(
				keywords,
				program.categories ? program.categories.join(' ') : '',
				program.searchCriteria || UserSettings.words
			);
		}

		// 配信者
		const author = row.querySelector('[itemprop="author"]');
		if (program.author) {
			const name = author.querySelector('[itemprop="name"]');
			const workLocation = author.querySelector('[itemprop="workLocation"]');
			if (!previousRow || program.author.name !== name.value
				|| program.author.url && program.author.url !== workLocation.href) {
				name.value = program.author.name;
				this.setMarkedText(workLocation, program.author.name);
				if (program.author.url) {
					workLocation.href = program.author.url;
				}
			}
		}

		// 説明文
		const description = row.querySelector('[itemprop="description"]');
		if (program.summary && !(previousRow && program.summary === description.value)) {
			description.value = program.summary;
			this.setMarkedText(description, program.summary, program.searchCriteria || UserSettings.words);
		}

		// 累計来場者数
		const userInteractionCount = row.querySelector('[itemprop="userInteractionCount"]');
		if (typeof program.visitors === 'number') {
			userInteractionCount.value = program.visitors;
			userInteractionCount.textContent = _('%d 人').replace('%d', program.visitors);
		}

		// コメント数
		const commentCount = row.querySelector('[itemprop="commentCount"]');
		if (typeof program.comments === 'number') {
			commentCount.value = program.comments;
			commentCount.textContent = _('%d コメ').replace('%d', program.comments);
		}

		// コミュニティ
		const productionCompany = row.querySelector('[itemprop="productionCompany"]');
		if (program.community) {
			productionCompany.querySelector('[itemprop="name"]').value = program.community.name;
			const url = productionCompany.querySelector('[itemprop="url"]');
			this.setMarkedText(url, program.community.name, program.searchCriteria || UserSettings.words);
			if (program.community.url) {
				url.href = program.community.url;
			}
		}

		// リンク
		for (const url of row.querySelectorAll(
			'td > [itemprop="url"], td:not([itemscope]) > [itemprop="name"] > [itemprop="url"]'
		)) {
			url.href = program.link;
		}

		return row;
	},

	/**
	 * 検索対象だった文字列を記入します。
	 * @param {HTMLElement} target
	 * @param {string} str
	 * @param {(SearchCriteria|SearchCriteria[])} [searchCriteria] - 検索対象の項目でない (文字数制限のみを行う) 場合は省略。
	 */
	setMarkedText: function (target, str, searchCriteria) {
		/**
		 * 昇順に並んでいる、重なる部分が無い [先頭のオフセット, 末尾のオフセット] の配列。
		 * @type {(number[])[]}
		 */
		const offsets = searchCriteria ? WordProcessor.getMatches(str, searchCriteria) : [];

		/**
		 * 挿入するTextノード。
		 * @type {Text}
		 */
		const text = new Text(str);

		/**
		 * mark要素に内包する範囲。
		 * @type {Range[]}
		 */
		const ranges = WordProcessor.convertOffsetsToRanges(text, offsets);

		if (str.length > UserSettings.MAX_VISIBLE_CHARACTERS
			&& document.getElementsByName('ellipsisTooLongRSSData')[0].checked) {
			// 文字数が制限を超えており、表示文字数の制限が有効なら
			target.classList.add(...WordProcessor.extractTextNode(
				text,
				offsets,
				UserSettings.MAX_VISIBLE_CHARACTERS,
				UserSettings.MAX_BEFORE_CHARACTERS)
			);
			const title = WordProcessor.markMatchesAsPlainText(str, offsets);
			if (/\(Windows .+ Chrome\/(?!.+Edge\/)/.test(navigator.userAgent)) {
				// Windows 版の Opera、Google Chrome におけるフリーズの回避
				target.dataset.title = title;
			} else {
				target.title = title;
			}
		} else {
			delete target.dataset.title;
			target.removeAttribute('title');
		}

		while (target.hasChildNodes()) {
			target.firstChild.remove();
		}
		target.append(text);
		for (const range of ranges) {
			range.surroundContents(document.createElement('mark'));
		}
	},

	/**
	 * 指定された列をキーに行を並べ替えます。
	 * @param {HTMLTableHeaderCellElement} th
	 */
	sort: function (th) {
		const table = th.closest('table');

		/**
		 * すでに並び替えられている列。
		 * @type {HTMLTableHeaderCellElement}
		 */
		const sortedTH = th.parentElement.querySelector('[data-sorted]');

		// 行リストを配列化
		const tBody = table.tBodies[0];
		const rows = Array.from(tBody.rows);

		if (sortedTH === th) {
			// 選択された列が、すでに並べ替えられている列なら
			// data-sorted属性の設定
			sortedTH.dataset.sorted = sortedTH.dataset.sorted === 'asc' ? 'desc' : 'asc';
			// 並び順を反転
			rows.reverse();
		} else {
			// 他の列のsorted属性を削除
			sortedTH.removeAttribute('data-sorted');

			// sorted属性の設定
			th.dataset.sorted = 'asc';

			// 昇順に並び替え
			rows.sort((a, b) => this.compareRows(th, a, b));
		}

		// 画面に反映
		tBody.removeAttribute('aria-live');
		tBody.append(...rows);
		window.setTimeout(function () {
			tBody.setAttribute('aria-live', 'polite');
		}, 0);

		// 並び順を保存
		GM.setValue('order', JSON.stringify({name: th.id, order: th.dataset.sorted}));
	},

	/**
	 * 行の挿入位置を取得します。
	 * @param {HTMLTableElement} table
	 * @param {HTMLTableRowElement} row
	 * @returns {number} 0から始まるインデックス。
	 */
	getInsertPosition: function (table, row) {
		const insertingColumn = table.querySelector('[data-sorted]');
		const reversed = insertingColumn.dataset.sorted === 'desc';
		const rows = table.tBodies[0].rows;
		let insertPosition = rows.length;
		for (const comparisonRow of Array.from(rows)) {
			const result = this.compareRows(insertingColumn, comparisonRow, row);
			if (reversed ? result < 0 : 0 < result) {
				insertPosition = comparisonRow.sectionRowIndex;
				break;
			}
		}
		return insertPosition;
	},

	/**
	 * {@link Array#sort}の比較関数内で用いる、行と行を比較する関数
	 * @param {HTMLTableHeaderCellElement} th - キーとなるセル。
	 * @param {HTMLTableRowElement} a
	 * @param {HTMLTableRowElement} b
	 * @returns {number} a < b なら -1、a > b なら 1 を返す
	 */
	compareRows: function (th, a, b) {
		return this.getCellValue(th, a) < this.getCellValue(th, b) ? -1 : 1;
	},

	/**
	 * セルの値を取得します。
	 * @param {HTMLTableHeaderCellElement} th - キーとなるセル。
	 * @param {HTMLTableRowElement} row
	 * @returns {string}
	 */
	getCellValue: function (th, row) {
		let value;
		const cell = row.cells[th.cellIndex];
		const child = cell.firstElementChild;
		switch (child && child.localName) {
			case 'data':
				value = child.value;
				if (/^[0-9]+$/.test(value)) {
					value = Number.parseInt(value);
				}
				break;
			case 'time':
				value = DateUtils.parseDurationString(child.dateTime);
				break;
			default:
				value = cell.textContent;
		}
		return value;
	},

	/**
	 * ステータス行を除くすべての行を取得します。
	 * @param {HTMLTableElement} table
	 * @returns {HTMLTableRowElement[]}
	 * @access private
	 */
	getRows: function (table) {
		const rows = Array.from(table.querySelectorAll(':not(tfoot) > tr'));
		rows.push(table.getElementsByTagName('template')[0].content.firstElementChild);
		return rows;
	},

	/**
	 * 列 th を列 refTH の前に移動します。
	 * @param {HTMLTableHeaderCellElement} th
	 * @param {?(HTMLTableHeaderCellElement|number)} refTH - null の場合、末尾に移動します。
	 */
	moveColumn: function (th, refTH) {
		const targetIndex = th.cellIndex;
		const refIndex = typeof refTH === 'number' ? refTH : (refTH ? refTH.cellIndex : -1);
		for (const tr of this.getRows(th.closest('table'))) {
			tr.insertBefore(tr.cells[targetIndex], tr.cells[refIndex]);
		}

		// 表示する列の設定項目の並び替え
		const ul = document.getElementById('visible-columns');
		ul.insertBefore(ul.querySelector(`[value="${th.id}"]`).closest('li'), ul.children[refIndex]);
	},

	/**
	 * 列 th を隠します。
	 * @param {HTMLTableHeaderCellElement} th
	 */
	hideColumn: function (th) {
		if (th.getAttribute('aria-hidden') !== 'true') {
			const targetIndex = th.cellIndex;
			for (const tr of this.getRows(th.closest('table'))) {
				tr.cells[targetIndex].setAttribute('aria-hidden', 'true');
			}
		}
	},

	/**
	 * 列 th を表示します。
	 * @param {HTMLTableHeaderCellElement} th
	 */
	showColumn: function (th) {
		if (th.getAttribute('aria-hidden') === 'true') {
			const targetIndex = th.cellIndex;
			for (const tr of this.getRows(th.closest('table'))) {
				tr.cells[targetIndex].removeAttribute('aria-hidden');
			}
		}
	},

	/**
	 * 列の順番を取得します。
	 * @returns {string[]}
	 */
	getColumnPositions: function () {
		return Array.from(document.querySelectorAll('#programs th')).map(function (th) {
			return th.id;
		});
	},

	/**
	 * 列の移動先を示すクラス名を削除します。
	 */
	removeOldClassName: function () {
		const oldRef = document.querySelector('.inserting-before, .inserting-after');
		if (oldRef) {
			oldRef.classList.remove('inserting-before', 'inserting-after');
		}
	},

	/**
	 * 列の順番を反映します。
	 * @param {?string} version
	 * @param {string[]} columnPositions - 指定されなかった列は末尾に並びます。
	 * @returns {Promise.<void>}
	 */
	reflectColumnPositions: async function (version, columnPositions) {
		if (!version) {
			// 5.0.0 より前のバージョンの設定であれば
			columnPositions.unshift('service');
			await GM.setValue('columns-position', JSON.stringify(columnPositions));
		}

		const tBody = document.querySelector('#programs tbody');
		tBody.removeAttribute('aria-live');
		let i = 0;
		for (const column of columnPositions) {
			this.moveColumn(document.getElementById(column), i);
			i++;
		}
		window.setTimeout(function () {
			tBody.setAttribute('aria-live', 'polite');
		}, 0);
	},
};

/**
 * ユーザー設定値、およびその変更に関するメソッド、プロパティ。
 */
const UserSettings = {
	/**
	 * 省略設定が有効の時に、一項目で表示する最大の符号単位数。
	 * @constant {number}
	 */
	MAX_VISIBLE_CHARACTERS: 60,

	/**
	 * 表示を省略した際に、ヒットした文字列より前に表示する最大の符号単位数。
	 * @constant {number}
	 */
	MAX_BEFORE_CHARACTERS: 3,

	/**
	 * 検索条件。
	 * @type {SearchCriteria[]}
	 */
	words: [],

	/**
	 * 検索から除外するユーザー、コミュニティ、チャンネルのURL。exclusionsFromExternalを含む。
	 * @type {string[]}
	 */
	exclusions: [],

	/**
	 * ユーザー設定値 NGsURI から取得した検索から除外するユーザー、コミュニティ、チャンネルのURL。
	 * @type {string[]}
	 */
	exclusionsFromExternal: [],

	/**
	 * ユーザー設定値をJSONファイルにエクスポートします。
	 */
	export: async function () {
		const exportedValues = {};
		for (const key in UserSettings.schema.properties) {
			const property = UserSettings.schema.properties[key];
			let value;
			switch (property.type) {
				case 'string':
				case 'integer':
				case 'boolean':
					value = key === 'audioData' ? await UserSettings.getLargeValue(key) : await GM.getValue(key);
					break;
				case 'number':
					value = await GM.getValue(key);
					if (value !== undefined && value !== null) {
						value = Number.parseFloat(value);
					}
					break;
				case 'array':
				case 'object':
					value = await GM.getValue(key);
					if (value !== undefined && value !== null) {
						value = JSON.parse(value);
					}
					break;
			}
			if (value !== undefined && value !== null) {
				exportedValues[key] = value;
			}
		}

		document.body.insertAdjacentHTML(
			'beforeend',
			h`<a href="${window.URL.createFor(
				new Blob([JSON.stringify(exportedValues, null, '\t')],{ type: 'application/json' })
			)}" download="${Alert.ID + '.json'}" hidden=""></a>`
		);
		const anchor = document.body.lastElementChild;
		anchor.click();
		anchor.remove();
	},

	/**
	 * ファイルのインポート用に生成したinput要素を取り除くまでのミリ秒数。
	 * @constant {number}
	 */
	MAX_LIFETIME: 10 * DateUtils.MINUTES_TO_MILISECONDS,

	/**
	 * ユーザー設定値をJSONファイルからインポートします。
	 */
	import: function () {
		document.body.insertAdjacentHTML(
			'beforeend',
			h`<input type="file" accept=".json,application/json" hidden="" />`
		);
		const input = document.body.lastElementChild;

		input.addEventListener('change', function parse(event) {
			event.target.removeEventListener(event.type, parse);
			const file = event.target.files[0];
			if (file) {
				const reader = new FileReader();
				reader.addEventListener('load', async function (event) {
					let result;
					try {
						result = JSON.parse(event.target.result);
					} catch (error) {
						window.alert(_('インポートに失敗しました。\n\nエラーメッセージ:\n%s').replace('%s', error));
					}

					if (result !== undefined) {
						if (Object.prototype.toString.call(result) === '[object Object]') {
							for (const key in result) {
								if (result[key] === null) {
									delete result[key];
								}
							}
						}
						const validate = jsen(UserSettings.schema, { greedy: true });
						if (validate(result)) {
							// 検索語句
							document.getElementById('searching-words').value
								= result.words ? result.words.join('\n') + '\n' : '';
							document.getElementsByName('save-searching-words')[0].click();

							// 除外リスト
							document.getElementById('ng-communities').value
								= result.NGs ? result.NGs.join('\n') + '\n' : '';
							document.getElementsByName('save-ng-communities')[0].click();

							// 外部の除外リストURL
							if (result.NGsURI) {
								await GM.setValue('NGsURI', result.NGsURI);
							} else {
								await GM.deleteValue('NGsURI');
							}

							// 行のソート
							if (!result.order) {
								result.order = UserSettings.schema.properties.order.default;
							}
							let sortedTH = document.querySelector('#programs [data-sorted]');
							if (result.order.name !== sortedTH.id) {
								sortedTH = document.getElementById(result.order.name);
								sortedTH.click();
							}
							if (result.order.order !== sortedTH.dataset.sorted) {
								sortedTH.click();
							}

							// 列の位置
							if (result['columns-position']) {
								await GM.setValue('columns-position', JSON.stringify(result['columns-position']));
								await TableProcessor.reflectColumnPositions(result.version, result['columns-position']);
							} else {
								await GM.deleteValue('columns-position');
								await TableProcessor.reflectColumnPositions(
									GM.info.script.version,
									UserSettings.schema.properties['columns-position'].default
								);
							}

							// 表示される列
							await UserSettings.showColumns(result.version, result['visible-columns']
								? result['visible-columns']
								: UserSettings.schema.properties['visible-columns'].default);

							// 検索対象のサービス
							UserSettings.enableServices(result.version, result['target-services'] || serviceIds);

							// プライベート配信・長い文字列の省略・言語
							for (const key of ['exclusionMemberOnly', 'ellipsisTooLongRSSData', 'languageFilter']) {
								const input = document.getElementsByName(key)[0];
								if (input.checked
									!== (key in result ? result[key] : UserSettings.schema.properties[key].default)) {
									input.click();
								}
							}

							// ミュート
							const alertTone = document.getElementById('alert-tone');
							if (result.audioMuted) {
								alertTone.muted = true;
							}

							// 音量
							if ('audioVolume' in result) {
								alertTone.volume = result.audioVolume;
							}

							// 音声ファイル
							const deleteSound = document.getElementsByName('delete-sound')[0];
							if (!deleteSound.hidden) {
								deleteSound.click();
							}
							if (result.audioData) {
								const audio = new Audio(result.audioData);
								audio.addEventListener('loadeddata', function () {
									if (audio.error) {
										// ブラウザが再生できないデータなら
										window.alert(
											_('使用中のブラウザが対応していないファイル形式のため、プロパティ %p を無視しました。').replace('%p', 'audioData')
										);
									} else {
										UserSettings.setAudioData(result.audioData);
									}
								});
								audio.addEventListener('error', function () {
									// ブラウザが再生できないデータなら
									window.alert(
										_('使用中のブラウザが対応していないファイル形式のため、プロパティ %p を無視しました。').replace('%p', 'audioData')
									);
								});
							}
						} else {
							window.alert(
								_('インポートに失敗しました。\n\nエラーメッセージ:\n%s')
									.replace('%s', JSON.stringify(validate.errors, null, '\t'))
							);
						}
					}
				});
				reader.readAsText(file);
			}
			input.remove();
		});
		input.click();

		window.setTimeout(function () {
			if (input.parentNode) {
				input.remove();
			}
		}, this.MAX_LIFETIME);
	},

	/**
	 * 音声ファイルを設定します。
	 * @param {string} audioData - data URL。
	 * @returns {Promise.<void>}
	 */
	setAudioData: async function (audioData) {
		try {
			await UserSettings.setLargeValue('audioData', audioData);
			const alertTone = document.getElementById('alert-tone');
			alertTone.src = audioData;
			alertTone.hidden = false;
			document.getElementsByName('delete-sound')[0].hidden = false;
		} catch (error) {
			if (error.name === 'QuotaExceededError' || error.name === 'NS_ERROR_DOM_QUOTA_REACHED') {
				window.alert(_('ファイルサイズが大きいため、設定に失敗しました。\n\nエラーメッセージ:\n%s').replace('%s', error));
			} else {
				throw error;
			}
		}
	},

	/**
	 * 検索語句文字列を {@link SearchCriteria} に変換します。
	 * @param {string[]} words - 正規化済みの文字列。
	 * @returns {SearchCriteria[]}
	 */
	parseWords: function (words) {
		return words.map(function (word) {
			const searchCriteria = {
				plus: [],
				minus: [],
			};
			for (const value of Normalizer.unifyCases(word).split(' ')) {
				if (value.startsWith('-')) {
					searchCriteria.minus.push(value.slice(1));
				} else {
					searchCriteria.plus.push(value);
				}
			}
			return searchCriteria;
		});
	},

	/**
	 * バージョン5.0.0より前の除外IDをURLにします。
	 * @param {string[]} exclusions
	 * @returns {string[]}
	 */
	parseExclusions: function (exclusions) {
		return exclusions.map(this.parseExclusion).filter(url => url);
	},

	/**
	 * バージョン5.0.0より前の除外IDをURLにします。
	 * @param {string} exclusion
	 * @returns {?string}
	 */
	parseExclusion: function (exclusion) {
		let url = null;
		if (exclusion.startsWith('http')) {
			url = exclusion;
		} else {
			const result = /(?:co|ch)[1-9][0-9]*/.exec(exclusion);
			if (result) {
				url = 'http://com.nicovideo.jp/community/' + result[0];
			}
		}
		return url;
	},

	/**
	 * 検索対象のサービスを取得します。
	 * @returns {string[]}
	 */
	getTargetServices: function () {
		return Array.from(document.querySelectorAll('[name="target-services"]:checked'))
			.map(checkbox => checkbox.value);
	},

	/**
	 * 表示中の列を取得します。
	 * @returns {string[]}
	 */
	getShownColumns: function () {
		return Array.from(document.querySelectorAll('[name="visible-columns"]:checked'))
			.map(checkbox => checkbox.value);
	},

	/**
	 * 指定されたサービスを有効化し、それ以外を無効化します。
	 * @param {?string} version
	 * @param {string[]} services
	 */
	enableServices: function (version, services) {
		if (compareVersions(version, '5.3.0') === -1) {
			// version < 5.3.0
			services.push('fresh', 'whowatch', 'periscope');
		}

		for (const service of document.getElementsByName('target-services')) {
			if ((services.indexOf(service.value) !== -1) !== service.checked) {
				service.click();
			}
		}
	},

	/**
	 * 最後に検索結果の取得に成功にした日時を表示します。
	 * @param {Service} service
	 */
	showLatestUpdatedDate: function (service) {
		const date = new Date();
		const html = h`
			<time datetime="${date.toISOString()}">
				${date.toLocaleString()}
			</time>
		`;
		document.querySelector(`[name="target-services"][value="${service.id}"]`).closest('tr').cells[1]
			.innerHTML = html;
		document.querySelector('#programs tfoot tr:nth-of-type(1) td').innerHTML = h(_('%s 更新')).replace('%s', html);
	},

	/**
	 * 直近の例外を表示します。
	 * @param {Service} service
	 * @param {Error} error
	 */
	showLatestError: function (service, error) {
		let message = error.toString();
		if ('lineNumber' in error) {
			message += ` (${error.lineNumber}:${error.columnNumber})`;
		}
		const html = h`<pre>${message}</pre>`;
		document.querySelector(`[name="target-services"][value="${service.id}"]`).closest('tr').cells[2]
			.innerHTML = html;
		document.querySelector('#programs tfoot tr:nth-of-type(2) td').innerHTML = html;
	},

	/**
	 * 指定された列を表示し、それ以外を非表示にします。
	 * @param {?string} version
	 * @param {string[]} columns
	 * @returns {Promise.<void>}
	 */
	showColumns: async function (version, columns) {
		if (!version) {
			// 5.0.0 より前のバージョンの設定であれば
			columns.unshift('service');
			if (columns.indexOf('category') === -1) {
				columns.push('category');
			}
			await GM.setValue('visible-columns', JSON.stringify(columns));
		}

		for (const column of document.getElementsByName('visible-columns')) {
			if ((columns.indexOf(column.value) !== -1) !== column.checked) {
				column.click();
			}
		}
	},

	/**
	 * Firefox 23 からの仕様変更 (Bug 836263) により、UserScriptLoader.uc.js において {@link GM_setValue} で1MiB以上の
	 * データを保存できなくなったため、容量制限を超過したデータはローカルストレージに保存します。
	 * @param {string} name
	 * @param {(string|number|boolean)} value
	 * @returns {Promise.<(string|number|boolean)>}
	 * @see [GM_setValue size exception(1 * 1024 * 1024) · Issue #1 · Constellation/ldrfullfeed · GitHub]{@link https://github.com/Constellation/ldrfullfeed/issues/1}
	 */
	setLargeValue: async function (name, value) {
		await GM.setValue(name, value);
		if (await GM.getValue(name) !== value) {
			// 値が正しく設定されていなければ
			const item = this.getValuesFromLocalStorage();
			item[name] = value;
			window.localStorage.setItem(Alert.ID, JSON.stringify(item));
			await GM.deleteValue(name);
		}
		return value;
	},

	/**
	 * {@link UserSettings.setLargeValue} で保存したデータを取得します。
	 * @param {type} name
	 * @param {*} defaultValue
	 * @returns {Promise}
	 */
	getLargeValue: async function (name, defaultValue) {
		let value = await GM.getValue(name);
		if (value === undefined || value === null) {
			const item = this.getValuesFromLocalStorage();
			value = item[name];
		}
		return value === undefined ? defaultValue : value;
	},

	/**
	 * {@link UserSettings.setLargeValue} で保存したデータを削除します。
	 * @param {string} name
	 */
	deleteLargeValue: function (name) {
		GM.deleteValue(name);
		const item = this.getValuesFromLocalStorage();
		delete item[name];
		window.localStorage.setItem(Alert.ID, JSON.stringify(item));
	},

	/**
	 * {@link UserSettings.setLargeValue} {@link UserSettings.getLargeValue} {@link UserSettings.deleteLargeValue}
	 * から利用される、すべての設定値を取得する関数。
	 * @returns {Object.<(string|number|boolean)>}
	 * @acsess private
	 */
	getValuesFromLocalStorage: function () {
		let item = window.localStorage.getItem(Alert.ID);
		if (item) {
			try {
				item = JSON.parse(item);
			} catch (e) {
				item = {};
			}
		} else {
			item = {};
		}
		return item;
	},
};






/**
 * アラートページのURL。
 * @type {string}
 */
const alertPageURL = 'http://live.nicovideo.jp/s/niconamaguide?' + Alert.ID;

/**
 * v5.3.3までのアラートページのURL。
 * @type {string}
 */
const oldAlertPageURL = 'http://live.nicovideo.jp/alert/?' + Alert.ID;

if (window.location.href === oldAlertPageURL) {
	window.location.replace(alertPageURL);
} else if (window.location.href !== alertPageURL && window.location.href !== alertPageURL.replace('http://', 'https://')) {
	GM.registerMenuCommand(Alert.NAME, function () {
		GM.openInTab(alertPageURL);
	});
} else {
	startScript(
		main,
		parent => parent.localName === 'body',
		target => target.id === 'utility_link',
		() => document.getElementById('utility_link')
	);
}

async function main()
{
	/**
	 * ライブストリーミング配信サービスのリスト。
	 * @type {Service[]}
	 */
	Alert.services = [new Service({
		id: 'fc2-live',
		name: _('FC2ライブ'),
		url: 'http://live.fc2.com/',
		disabledSearch: true,
		disabledLanguageFilter: true,
		getDetails: function () {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'https://live.fc2.com/contents/allchannellist.php',
				responseType: 'json',
			};
		},
		parseResponse: function (response) {
			return { programs: response.channel.filter(function (program) {
				return program.type !== 2 && program.login !== 2 && program.pay === 0;
			}) };
		},
		convertIntoEntry: function (program) {
			const url = 'http://live.fc2.com/' + program.id;
			return new Program(url, program.title, {
				icon: program.image,
				published: new Date(program.start.replace(' ', 'T') + '+09:00'),
				categories: program.category === 0
					? null
					: [['雑談', 'ゲーム', '作業', '動画', 'その他', null, '公式'][program.category - 1]],
				author: {
					name: program.name,
					url: url,
				},
				visitors: Number.parseInt(program.total),
				language: program.lang,
			});
		},
	}), new Service({
		id: 'cavetube',
		name: _('CaveTube'),
		url: 'https://gae.cavelis.net/',
		disabledSearch: true,
		getDetails: function () {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'https://rss.cavelis.net/index_live.xml',
				responseType: 'document',
			};
		},
		parseResponse: function (response) {
			return { programs: response.getElementsByTagNameNS(MarkupUtils.ATOM_NAMESPACE, 'entry') };
		},
		/**
		 * CaveTube名前空間。
		 * @constant {string}
		 */
		CT_NAMESPACE: 'http://gae.cavelis.net',
		convertIntoEntry: function (program) {
			const name = program.getElementsByTagNameNS(MarkupUtils.ATOM_NAMESPACE, 'name')[0].textContent;
			const parser = new DOMParser();
			let icon = program.getElementsByTagNameNS(this.CT_NAMESPACE, 'thumbnail_path')[0].textContent;
			if (icon === 'http:/img/no_thumbnail_image.png') {
				icon = 'https://www.cavelis.net/img/no_thumbnail_image.png';
			} else {
				const url = new URL(icon);
				url.protocol = 'https';
				icon = url.href;
			}
			return new Program(
				program.getElementsByTagNameNS(MarkupUtils.ATOM_NAMESPACE, 'link')[0].getAttribute('href'),
				parser.parseFromString(
					program.getElementsByTagNameNS(MarkupUtils.ATOM_NAMESPACE, 'title')[0].textContent, 'text/html'
				).body.textContent,
				{
					icon: icon,
					published: new Date(
						program.getElementsByTagNameNS(MarkupUtils.ATOM_NAMESPACE, 'published')[0].textContent
					),
					author: {
						name: name,
						url: 'https://gae.cavelis.net/user/' + encodeURIComponent(name),
					},
					categories: program.getElementsByTagNameNS(this.CT_NAMESPACE, 'tag')[0].textContent.split(' '),
					summary: parser.parseFromString(
						program.getElementsByTagNameNS(MarkupUtils.ATOM_NAMESPACE, 'summary')[0].textContent,
						'text/html'
					).body.textContent,
					visitors: Number.parseInt(
						program.getElementsByTagNameNS(this.CT_NAMESPACE, 'viewer')[0].textContent
					),
					comments: Number.parseInt(
						program.getElementsByTagNameNS(this.CT_NAMESPACE, 'comment_num')[0].textContent
					),
				}
			);
		},
	}), new Service({
		id: 'showroom',
		name: _('SHOWROOM'),
		url: 'https://www.showroom-live.com/',
		disabledSearch: true,
		getDetails: function () {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'https://www.showroom-live.com/onlive',
				responseType: 'document',
			};
		},
		parseResponse: function (response) {
			return { programs: response.getElementsByClassName('onlive-list-li') };
		},
		convertIntoEntry: function (program) {
			return new Program(
				program.getElementsByClassName('overview-link')[0].href,
				program.getElementsByClassName('tx-title')[0].textContent,
				{
					icon: program.getElementsByClassName('img-main')[0].dataset.src,
					published: DateUtils.parseJSTString(program.getElementsByClassName('time')[0].textContent),
					visitors: Number.parseInt(program.getElementsByClassName('view')[0].textContent),
				}
			);
		},
	}), new Service({
		id: 'stickam-japan',
		name: _('Stickam JAPAN!'),
		url: 'https://www.stickam.jp/',
		disabledSearch: true,
		getDetails: function () {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'https://www.stickam.jp/explore/session?page=1',
				responseType: 'document',
			};
		},
		parseResponse: function (response, details) {
			const programs = {
				programs: Array.from(response.getElementsByClassName('col-md-3 col-sm-3'))
					.filter(program => !program.querySelector('#ad-name')),
			};
			const nextPage = response.querySelector('.pagination > li:last-of-type:not(.disabled) a');
			if (nextPage) {
				details.url = nextPage.href;
				programs.next = details;
			}
			return programs;
		},
		convertIntoEntry: function (program) {
			const userNameLink = program.getElementsByTagName('a')[0];
			return new Program(userNameLink.href, userNameLink.text, {
				icon: program.getElementsByClassName('embed-responsive-item')[0],
				private: Boolean(program.getElementsByClassName('status')[0]),
				published: DateUtils.parseJSTString(
					program.getElementsByClassName('post-info')[0].textContent.replace('~', '')
				),
				author: {
					name: userNameLink.text,
					url: userNameLink.href.replace('stickon#webcam', ''),
				},
				summary: program.getElementsByTagName('p')[0].textContent,
			});
		},
	}), new Service({
		id: 'twitcasting',
		name: _('ツイキャス'),
		url: 'https://twitcasting.tv/',
		disabledOr: true,
		disabledMinus: true,
		disabledLanguageFilter: true,
		getDetails: function (searchCriteria) {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'https://twitcasting.tv/search/text/' + encodeURIComponent(searchCriteria.plus.join(' ')),
				responseType: 'document',
			};
		},
		parseResponse: function (response) {
			return { programs: response.querySelectorAll('#content > div:first-of-type td') };
		},
		convertIntoEntry: function (program) {
			const url = program.querySelector('.searcheduser a').href;
			const titleAndComments = /(.*?)(?: \((0|[1-9][0-9]*)\))?$/.exec(program.querySelector('.title a').text);
			const countryflag = program.getElementsByClassName('countryflag')[0];
			let language;
			switch (countryflag && countryflag.src) {
				case 'https://twitcasting.tv/img/c/us.gif':
					language = 'en';
					break;
				case 'https://twitcasting.tv/img/c/br.gif':
					language = 'pt';
					break;
				case 'https://twitcasting.tv/img/c/mx.gif':
					language = 'es';
					break;
				case 'https://twitcasting.tv/img/c/jp.gif':
					language = 'ja';
					break;
				default:
					language = 'und';
			}
			return new Program(url, titleAndComments[1], {
				icon: program.getElementsByClassName('icon32')[0].src,
				author: {
					name: program.getElementsByClassName('fullname')[0].textContent,
					url: url,
				},
				categories: Array.from(program.getElementsByClassName('tag'), function (anchor) {
					return anchor.text;
				}),
				summary: program.getElementsByClassName('userdesc')[0].textContent.trim(),
				comments: titleAndComments[2] ? Number.parseInt(titleAndComments[2]) : null,
				language: language,
			});
		},
	}), new Service({
		id: 'twitch',
		name: _('Twitch'),
		url: 'https://www.twitch.tv/',
		/**
		 * 検索結果の最大件数。
		 * @constant {number}
		 */
		MAX_RESULT_LENGTH: 100,
		/**
		 * 本スクリプトのクライアントID。
		 * @constant {string}
		 */
		CLIENT_ID: '240onlpbs6y5fwt92jj26i8bb4inszw',
		disabledOr: true,
		disabledMinus: true,
		disabledLanguageFilter: true,
		getDetails: function (searchCriteria) {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'https://api.twitch.tv/kraken/search/streams?' + new URLSearchParams(
					{client_id: this.CLIENT_ID, limit: this.MAX_RESULT_LENGTH, q: searchCriteria.plus.join(' ')}
				),
				responseType: 'json',
			};
		},
		parseResponse: function (response) {
			return { programs: response.streams };
		},
		convertIntoEntry: function (program) {
			return new Program(program.channel.url, program.channel.status || 'Untitled Broadcast', {
				icon: program.channel.logo || program.channel.profile_banner,
				published: new Date(program.created_at),
				author: {
					name: program.channel.display_name,
					url: program.channel.url + '/profile',
				},
				categories: program.channel.game ? [program.channel.game] : null,
				visitors: program.viewers,
				language: program.channel.language,
			});
		},
	}), new Service({
		/**
		 * 検索結果の最大件数。
		 * @constant {number}
		 */
		MAX_RESULT_LENGTH: 100,
		id: 'niconico-live',
		name: _('ニコニコ生放送'),
		url: 'http://live.nicovideo.jp/',
		icon: 'http://nl.simg.jp/public/inc/assets/zero/img/base/favicon.ico',
		getDetails: function (words) {
			return {
				mode: 'no-cors',
				method: 'GET',
				responseType: 'json',
				url: 'http://api.search.nicovideo.jp/api/v2/live/contents/search?' + new URLSearchParams({
					q: words.map(searchCriteria => searchCriteria.plus[0]).join(' OR '),
					targets: ['title', 'tags', 'description'].join(),
					fields: [
						'contentId',        // ID
						'title',            // タイトル
						'memberOnly',       // プライベート
						'startTime',        // 開始時刻
						'communityIcon',    // コミュニティアイコン
						'tags',             // タグ
						'description',      // 詳細
						'communityId',      // コミュニティID
						'channelId',
						'viewCounter',
						'commentCounter',
					].join(),
					'filters[liveStatus][0]': 'onair',
					_sort: 'startTime',
					_limit: this.MAX_RESULT_LENGTH,
					_context: 'ニコ生アラート(簡)',
				}),
			};
		},
		parseResponse: function (response, details) {
			return { programs: response.data };
		},
		convertIntoEntry: function (program) {
			const otherDetails = {
				icon: program.communityIcon,
				private: Boolean(program.memberOnly),
				published: new Date(program.startTime),
				// <br /> タグを半角スペースに置き換え、他のタグは取り除く
				summary: program.description.replace(/<br( )\/>|<font[^>]+>|<\/?(?:font|b|i|s|u)>/g, '$1'),
				categories: program.tags && program.tags.split(' '),
				visitors: program.viewCounter,
				comments: program.commentCounter,
			};

			if (otherDetails.summary.includes('&')) {
				// 文字参照が含まれていれば
				otherDetails.summary
					= new DOMParser().parseFromString(otherDetails.summary, 'text/html').body.textContent;
			}

			if (program.communityId || program.channelId) {
				otherDetails.community = {
					name: program.communityId ? 'co' + program.communityId : 'ch' + program.channelId,
				};
				otherDetails.community.url = 'http://com.nicovideo.jp/community/' + otherDetails.community.name;
			}

			return new Program('http://live.nicovideo.jp/watch/' + program.contentId, program.title, otherDetails);
		},
		delay: 1 * DateUtils.MINUTES_TO_MILISECONDS,
	}), new Service({
		id: 'himawari-stream',
		name: _('ひまわりストリーム'),
		url: 'http://himast.in/',
		disabledSearch: true,
		getDetails: function () {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'http://himast.in/?mode=program&cat=search&sort=st_start_date&st_status=1&rss=1',
				responseType: 'document',
			};
		},
		parseResponse: function (response) {
			return { programs: response.getElementsByTagName('item') };
		},
		convertIntoEntry: function (program) {
			const details = [];
			const description = new DOMParser().parseFromString(
				program.getElementsByTagName('description')[0].textContent,
				'text/html'
			);
			let summary;
			for (const node of description.getElementsByClassName('riRssContributor')[0].childNodes) {
				if (node.nodeType === Node.TEXT_NODE) {
					details.push([node.data.replace(/^\s+|:|:/g, '')]);
				} else if (node.localName === 'b') {
					details[details.length - 1][1] = node.textContent;
				} else {
					summary = node.nextSibling.data;
					break;
				}
			}

			return new Program(
				program.getElementsByTagName('link')[0].textContent,
				program.getElementsByTagName('title')[0].textContent,
				{
					icon: description.getElementsByTagName('img')[0].src,
					published: new Date(program.getElementsByTagName('pubDate')[0].textContent),
					author: {
						name: details.find(function (detail) {
							return detail[0] === '配信者';
						})[1],
					},
					summary: summary,
					visitors: Number.parseInt(details.find(function (detail) {
						return detail[0] === '延べ入場者数';
					})[1]),
					comments: Number.parseInt(details.find(function (detail) {
						return detail[0] === 'コメント数';
					})[1]),
				}
			);
		},
	}), new Service({
		id: 'fresh',
		name: _('FRESH! (AbemaTV)'),
		url: 'https://freshlive.tv/',
		icon: 'https://freshlive.tv/assets/1482283729/favicon.ico',
		/**
		 * 検索結果の最大件数。
		 * @constant {number}
		 */
		MAX_RESULT_LENGTH: 1000,
		disabledOr: true,
		disabledMinus: true,
		getDetails: function (searchCriteria) {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'https://freshlive.tv/proxy/Searches;type=onairs'
					+ `;count=${this.MAX_RESULT_LENGTH};q=${encodeURIComponent(searchCriteria.plus.join(' '))}`,
				responseType: 'json',
			};
		},
		parseResponse: function (response) {
			return { programs: response.data.filter(program => program.paymentStatus === 'free') };
		},
		convertIntoEntry: function (program) {
			return new Program(program.permalink, program.title, {
				icon: program.channel.imageUrl,
				published: new Date(program.startAt),
				author: {
					name: program.channel.user.displayName,
				},
				categories: program.tags,
				summary: program.description,
				visitors: program.viewCount,
				comments: program.commentCount,
				community: {
					name: program.channel.title,
					url: program.channel.permalink,
				},
			});
		},
		delay: 1 * DateUtils.MINUTES_TO_MILISECONDS,
	}), new Service({
		id: 'whowatch',
		name: _('ふわっち'),
		url: 'https://whowatch.tv/',
		icon: 'https://whowatch.tv/image/favicon.ico',
		disabledSearch: true,
		getDetails: function () {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'https://api.whowatch.tv/lives?category_id=0&list_type=new',
				responseType: 'json',
			};
		},
		parseResponse: function (response) {
			return { programs: response[0].new };
		},
		convertIntoEntry: function (program) {
			return new Program('https://whowatch.tv/viewer/' + program.id, program.title, {
				icon: program.user.icon_url,
				published: new Date(program.started_at),
				author: {
					name: program.user.name,
					url: 'https://whowatch.tv/profile/' + program.user.user_path,
				},
				visitors: program.total_view_count,
			});
		},
	}), new Service({
		id: 'periscope',
		name: _('Periscope (Twitter)'),
		url: 'https://www.periscope.tv/',
		disabledOr: true,
		disabledMinus: true,
		disabledLanguageFilter: true,
		getDetails: function (searchCriteria) {
			return {
				mode: 'no-cors',
				method: 'POST',
				url: 'https://api.periscope.tv/api/v2/broadcastSearchPublic',
				responseType: 'json',
				data: {search: searchCriteria.plus.join(' '), include_replay: false},
			};
		},
		parseResponse: function (response) {
			return { programs: response };
		},
		convertIntoEntry: function (program) {
			return new Program(`https://www.periscope.tv/${program.username}/${program.id}`, program.status, {
				icon: program.profile_image_url,
				published: new Date(program.start),
				author: {
					name: program.user_display_name,
					url: program.twitter_username && 'https://twitter.com/' + program.twitter_username,
				},
				categories: program.tags,
				visitors: program.n_total_watching,
				language: program.language,
			});
		},
		delay: 1 * DateUtils.MINUTES_TO_MILISECONDS,
	}), new Service({
		id: 'youtube-live',
		name: _('YouTube ライブ'),
		url: 'https://www.youtube.com/live',
		icon: 'https://i.ytimg.com/i/4R8DWoMoI7CAwX8_LjQHig/mq1.jpg',
		disabledOr: true,
		getDetails: function (word) {
			let query = word.plus.join(' ');
			if (word.minus.length > 0) {
				query += ' -' + word.minus.join(' -');
			}
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'https://www.youtube.com/results?'
					+ new URLSearchParams({filters: 'live', search_sort: 'video_date_uploaded', search_query: query}),
				responseType: 'document',
			};
		},
		parseResponse: function (response) {
			return { programs: response.getElementsByClassName('yt-lockup-dismissable') };
		},
		convertIntoEntry: function (program) {
			const anchor = program.querySelector('.yt-lockup-title a');
			const userAnchor = program.querySelector('.yt-lockup-byline a');
			let description = program.getElementsByClassName('yt-lockup-description')[0];
			const metaInfo = document.getElementsByClassName('yt-lockup-meta-info')[0];
			return new Program(anchor.href, anchor.title, {
				icon: program.querySelector('.yt-thumb img').src,
				author: {
					name: userAnchor.text,
					url: userAnchor.href,
				},
				summary: description ? description.textContent : null,
				visitors: metaInfo ? Number.parseInt(/[0-9]+/.exec(metaInfo)[0]) : null,
			});
		},
	}), new Service({
		/**
		 * 検索結果の最大件数。
		 * @constant {number}
		 */
		MAX_RESULT_LENGTH: 100,
		id: 'younow',
		name: _('YouNow'),
		url: 'https://www.younow.com/',
		disabledMinus: true,
		getDetails: function (words) {
			return {
				mode: 'cors',
				method: 'POST',
				url: 'https://qz0xcgubgq.algolia.io/1/indexes/*/queries',
				responseType: 'json',
				headers: {
					'X-Algolia-Application-Id': 'QZ0XCGUBGQ',
					'X-Algolia-API-Key': '7f270d4586d986ef69fb5bab5ecd7f741b5cb3f7042881112ed46c97b5e8404a',
					'X-Algolia-TagFilters': '(public)',
				},
				data: { requests: words.map(searchCriteria => ({
					indexName: 'people_search_live',
					params: new URLSearchParams(
						{hitsPerPage: this.MAX_RESULT_LENGTH, advancedSyntax: 1, query: searchCriteria.plus.join(' ')}
					).toString(),
				})) },
			};
		},
		parseResponse: function (response, details, words) {
			return { programs: Array.prototype.concat.apply([], response.results.map(function (result, index) {
				const programs = [];
				const searchCriteria = words[index];
				for (const program of result.hits) {
					if (program.tag === '') {
						break;
					}
					program.searchCriteria = searchCriteria;
					programs.push(program);
				}
				return programs;
			})) };
		},
		convertIntoEntry: function (program) {
			const url = 'https://www.younow.com/' + program.profile;

			return new Program(url, '#' + program.tag, {
				icon: 'https://cdn2.younow.com/php/api/channel/getImage/channelId=' + program.objectID,
				author: {
					name: program.firstName + ' ' + program.lastName,
					url: url,
				},
				categories: [program.tag],
				summary: program.description,
			}, program.searchCriteria);
		},
	}), new Service({
		id: 'livestream',
		name: _('Livestream'),
		url: 'https://livestream.com/watch/',
		icon: 'https://cdn.livestream.com/website/b0c04ce/assets/favicon.ico',
		/**
		 * 検索結果の最大件数。
		 * @constant {number}
		 */
		MAX_RESULT_LENGTH: 1000,
		getDetails: function (words) {
			return {
				mode: 'cors',
				method: 'POST',
				url: 'https://7kjecl120u-1.algolia.io/1/indexes/*/queries',
				responseType: 'json',
				data: {
					apiKey: '98f12273997c31eab6cfbfbe64f99d92',
					appID: '7KJECL120U',
					requests: words.map(searchCriteria => {
						let query = searchCriteria.plus.join(' ');
						if (searchCriteria.minus.length > 0) {
							query += ' -' + searchCriteria.minus.join(' -');
						}
						return {
							indexName: 'events',
							params: new URLSearchParams({
								hitsPerPage: this.MAX_RESULT_LENGTH,
								advancedSyntax: 1,
								facets: '*',
								facetFilters: '["is_live:1"]',
								query: query,
							}).toString(),
						};
					}),
				},
			};
		},
		parseResponse: function (response, details, words) {
			return { programs: Array.prototype.concat.apply([], response.results.map(function (result, index) {
				const searchCriteria = words[index];
				for (const program of result.hits) {
					program.searchCriteria = searchCriteria;
				}
				return result.hits;
			})) };
		},
		convertIntoEntry: function (program) {
			const tags = [];
			if (program.category_name !== 'No category') {
				tags.push(program.category_name);
			}
			if (program.subcategory_name) {
				tags.push(program.subcategory_name);
			}
			if (program.tags) {
				tags.concat(program.tags.split(','));
			}

			return new Program('https://livestream.com' + program.path, program.full_name, {
				icon: program.owner_logo ? program.owner_logo.thumbnail.url : null,
				private: Boolean(program.is_password_protected),
				published: new Date(program.start_time),
				author: {
					name: program.owner_account_full_name,
					url: 'https://livestream.com/accounts/' + program.owner_account_id,
				},
				categories: tags.length > 0 ? tags : null,
				visitors: program.concurrent_viewers_count,
				comments: program.live_video_post_comments_count,
			}, program.searchCriteria);
		},
	})];

	// ページタイトル
	document.title = Alert.NAME;

	// Favicon
	const icon = document.querySelector('[rel="icon"]');
	icon.href = Alert.ICON;
	document.head.append(icon);

	// 元のページ内容を削除
	document.getElementById('main_area').remove();

	// ヘッダを修正
	for (const select of document.querySelectorAll('[href^="javascript:"]')) {
		select.target = '_self';
	}

	// リンク先を新しいタブで開く、スタイルの設定
	document.head.insertAdjacentHTML('beforeend', h`
		<base target="_blank" />

		<style>
			[aria-hidden="true"] {
				display: none;
			}

			main {
				text-align: center;
				margin: 1em;
				/* フッターとブラウザ表示領域下端の隙間埋め */
				min-height: calc(100vh - (2em /* mainの上下マージン */ + ${document.body.clientHeight}px));
			}
			main a:link {
				color: mediumblue;
			}
			main a:visited {
				color: midnightblue;
			}

			/*====================================
				表
			 */
			#programs {
				width: 100%;
			}
			#programs caption {
				display: none;
			}
			main tr {
				background: silver;
				border-width: 1px;
				border-style: solid none;
			}
			main thead th {
				white-space: nowrap;
			}

			#programs tbody {
				text-align: left;
				border-top: solid;
				border-bottom: solid;
			}

			/*------------------------------------
				ボタンの左右のマージン
			 */
			main button {
				margin-left: 0.2em;
				margin-right: 0.2em;
			}

			/*------------------------------------
				セル内容の右寄せ・改行禁止
			 */
			#programs [role="timer"],
			#programs [aria-live="off"] {
				text-align: right;
				white-space: nowrap;
			}

			/*------------------------------------
				行の背景色
			 */
			#programs tbody tr:nth-of-type(2n-1),
			#user-setting-options tbody tr:nth-of-type(2n-1),
			#programs tbody:not(.odd) + tfoot tr {
				background: white;
			}
			main tr td,
			main tr th {
				padding: 3px;
			}

			/*------------------------------------
				リンクでない文字列
			 */
			a:not(:link) {
				color: unset;
				text-decoration: unset;
			}

			/*------------------------------------
				番組のアイコン
			 */
			[itemtype="http://schema.org/VideoObject"] [itemprop="image"] {
				width: 64px;
			}

			/*------------------------------------
				取得失敗 / 空文字列 / 空白文字
			 */
			#programs .illegal {
				font-style: oblique;
			}

			/*------------------------------------
				項目の移動
			 */
			.inserting-before {
				border-left: solid lightskyblue thick;
			}
			.inserting-after {
				border-right: solid lightskyblue thick;
			}
			#main-settings {
				-moz-column-count: 2;       /* Firefox */
				column-count: 2;
			}
			#main-settings textarea {
				width: 100%;
				height: 20em;
			}

			/*------------------------------------
				検索語句の強調表示
			 */
			main mark {
				color: inherit;
				background: khaki;
			}

			/*------------------------------------
				省略記号の表示
			 */
			.ellipsis-left::before,
			.ellipsis-right::after {
				content: "…";
				color: dimgray;
			}
			.ellipsis-left::before {
				margin-right: 0.2em;
			}
			.ellipsis-right::after {
				margin-left: 0.2em;
			}
			.ellipsis-left::before {
				margin-right: 0.2em;
			}
			.ellipsis-right::after {
				margin-left: 0.2em;
			}

			/*------------------------------------
				Windows 版の Opera、Google Chrome における全文の表示
			 */
			main td {
				position: relative;
			}
			[data-title]:hover::after {
				content: attr(data-title);
				position: absolute;
				top: calc(100% + 0.2em);
				left: 1em;
				border: 1px solid dimgray;
				background: khaki;
				padding: 0.5em;
				opacity: 0.9;
				border-radius: 0.7em;
				z-index: 1;
			}

			/*------------------------------------
				ステータス
			 */
			#programs tfoot tr:first-of-type {
				border-bottom: none;
			}
			#programs tfoot tr:last-of-type {
				border-top: none;
			}

			/*------------------------------------
				ライブ配信サービス一覧
			 */
			#user-setting-options tr {
				background: whitesmoke;
			}
			#user-setting-options thead th {
				padding-right: 1em;
			}

			/*====================================
				追加設定ボックス
			 */
			#user-setting-options {
				display: flex;
				width: 100%;
				flex-direction: column;
				align-items: center;
			}
			#user-setting-options dt {
				margin-top: 2em;
				font-weight: bold;
			}
			#user-setting-options dd > * {
				text-align: left;
			}
			#user-setting-options > :last-of-type li {
				list-style: disc;
			}

			/*------------------------------------
				▲▼
			 */
			[aria-controls="user-setting-options"]::before {
				content: "▼";
				color: darkslategray;
				margin-right: 0.5em;
			}
			[aria-controls="user-setting-options"][aria-expanded="true"]::before {
				content: "▲";
			}
			#user-setting-options[aria-hidden="true"] {
				display: none;
			}

			/*====================================
				ライブ配信サービスのアイコン
			 */
			[itemtype="http://schema.org/Organization"] [itemprop="logo"],
			[itemtype="http://schema.org/BroadcastService"] [itemprop="image"] {
				width: 16px;
			}
		</style>
	`);

	// 挿入
	document.getElementById('utility_link').insertAdjacentHTML('beforebegin', h`<main id="${Alert.ID}">
		<table id="programs">
			<caption>${_('検索ワードにヒットしたライブ配信番組')}</caption>
			<thead>
				<tr dropzone="move">
					<th id="service" title="${_('どのライブ配信サービスか')}" draggable="true">
					</th>
					<th id="thumbnail" title="${_('アイコン')}" draggable="true">
					</th>
					<th id="member_only" title="${_('プライベート配信か否か')}" draggable="true">
					</th>
					<th id="pubDate" title="${_('配信開始からの経過時間')}" data-sorted="asc" draggable="true">
						${_('経過')}
					</th>
					<th id="title" title="${_('番組のタイトル')}" draggable="true">
						${_('タイトル')}
					</th>
					<th id="category" title="${_('カテゴリ・タグ')}" draggable="true">
						${_('タグ')}
					</th>
					<th id="owner_name" title="${_('配信者の名前')}" draggable="true">
						${_('配信者')}
					</th>
					<th id="description" title="${_('説明文')}" draggable="true">
						${_('説明文')}
					</th>
					<th id="view" title="${_('来場者数')}" draggable="true">
						${_('来場')}
					</th>
					<th id="num_res" title="${_('総コメント数')}" draggable="true">
						${_('コメ数')}
					</th>
					<th id="community_name" title="${_('コミュニティ・チャンネル')}" draggable="true">
						${_('コミュニティ')}
					</th>
				</tr>
			</thead>
			<tbody aria-live="polite" aria-relevant="additions">
				<template>
					<tr itemscope="" itemtype="http://schema.org/VideoObject">
						<td itemprop="provider" itemscope="" itemtype="http://schema.org/Organization">
							<data itemprop="name" value="">
								<a itemprop="url">
									<img itemprop="logo" src="dummy" />
								</a>
							</data>
						</td>
						<td>
							<a itemprop="url"><img itemprop="image" src="dummy" hidden="" /></a>
						</td>
						<td>
							<data itemprop="requiresSubscription" value="false"></data>
						</td>
						<td role="timer">
							<time itemprop="duration" datetime="P365D" hidden=""></time>
						</td>
						<td>
							<data itemprop="name" value="">
								<a itemprop="url"></a>
							</data>
						</td>
						<td>
							<data itemprop="keywords" value="">
							</data>
						</td>
						<td itemprop="author" itemscope="" itemtype="http://schema.org/Person">
							<data itemprop="name" value="">
								<a itemprop="workLocation"></a>
							</data>
						</td>
						<td>
							<data itemprop="description" value="">
							</data>
						</td>
						<td aria-live="off" itemprop="interactionStatistic" itemscope=""
							itemtype="http://schema.org/InteractionCounter">
							<data itemprop="userInteractionCount" value="-1">
							</data>
						</td>
						<td aria-live="off">
							<data itemprop="commentCount" value="-1"></data>
						</td>
						<td itemprop="productionCompany" itemscope="" itemtype="http://schema.org/PerformingGroup">
							<data itemprop="name" value="">
								<a itemprop="url"></a>
							</data>
						</td>
					</tr>
				</template>
			</tbody>
			<tfoot>
				<tr>
					<td colspan="11" role="status"></td>
				</tr>
				<tr>
					<td colspan="11" role="status"></td>
				</tr>
			</tfoot>
		</table>
	</main>`);

	/**
	 * 番組を表示する表。
	 * @type {HTMLTableElement}
	 */
	const table = document.getElementById('programs');

	document.head.insertAdjacentHTML('beforeend', `<style>
		/*====================================
			列ごとの並べ替え
		 */
		#programs thead th:hover,
		#programs thead th:focus {
			cursor: pointer;
			background: gainsboro;
		}

		/*------------------------------------
			▲▼
		 */
		[data-sorted]::before {
			content: "▲";
			color: darkslategray;
			margin-right: 0.5em;
		}
		[data-sorted="desc"]::before {
			content: "▼";
		}
	</style>`);

	for (const th of Array.from(table.getElementsByTagName('th'))) {
		th.setAttribute('role', 'button');
		th.tabIndex = 0;
	}

	table.tHead.addEventListener('keydown', function (event) {
		if (event.key === ' ') {
			event.preventDefault();
		}
	});

	table.tHead.addEventListener('keyup', function (event) {
		if (event.key === 'Enter' || event.key === ' ') {
			TableProcessor.sort(event.target);
			event.target.blur();
		}
	});

	/**
	 * アプリケーション全体を内包する main 要素。
	 * @type {HTMLElement}
	 */
	const main = document.getElementById(Alert.ID);

	main.insertAdjacentHTML('beforeend', h`
		<dl id="main-settings">
			<dt>
				${_('検索語句')}
				<button name="save-searching-words" aria-controls="searching-words">${_('保存')}</button>
			</dt>
			<dd>
				<textarea name="searching-words" id="searching-words"></textarea>
			</dd>
			<dt>
				${_('除外するコミュニティ・チャンネルなどの URL')}
				<button name="save-ng-communities" aria-controls="ng-communities">${_('保存')}</button>
				<button name="sets-blacklist-uri">${_('除外 URL リストの取得先を設定')}</button></dt>
			<dd>
				<textarea name="ng-communities" id="ng-communities"></textarea>
			</dd>
		</dl>

		<audio id="alert-tone" controls="" preload="auto" hidden=""></audio>

		<button type="button" aria-expanded="false" aria-controls="user-setting-options">
			${_('追加設定ボックスの開閉')}
		</button>
		<dl id="user-setting-options" aria-hidden="true">
			<dt>${_('検索対象のライブ配信サービス')}</dt>
			<dd>
				<table>
					<thead>
						<tr>
							<th>${_('サービス名')}</th>
							<th>${_('最後に検索結果の取得に成功にした日時')}</th>
							<th>${_('直近のエラー')}</th>
						</tr>
					</thead>
					<tbody>` + Alert.services.map(function (service) {
						return h`<tr itemscope="" itemtype="http://schema.org/BroadcastService">
							<td>
								<label itemprop="name">
									<input name="target-services" value="${service.id}" type="checkbox">
									<img itemprop="image" src="${service.icon}" alt="" />
									${service.name}
								</label>
							</td>
							<td role="status">
							</td>
							<td role="status">
							</td>
						</tr>`;
					}).join('') + h`</tbody>
				</table>
			</dd>

			<dt>${_('表示する項目の設定')}</dt>
			<dd>
				<ul id="visible-columns">` + Array.from(main.getElementsByTagName('th')).map(function (th) {
					return h`<li>
						<label>
							<input name="visible-columns" value="${th.id}" aria-controls="${th.id}" type="checkbox"
								checked="">
							${_(th.title)}
						</label>
					</li>`;
				}).join('') + h`</ul>
			</dd>

			<dt>${_('その他の設定')}</dt>
			<dd>
				<ul>
					<li>
						<label>
							<input name="exclusionMemberOnly" type="checkbox">
							${_('プライベート配信を通知しない')}
						</label>
					</li>
					<li>
						<label>
							<input name="ellipsisTooLongRSSData" type="checkbox" checked="">
							${_('タイトル・キャプション・コミュニティ名が %d 文字を超えたら省略する')
								.replace('%d', UserSettings.MAX_VISIBLE_CHARACTERS)}
						</label>
					</li>
					<li>
						<label>
							<input name="languageFilter" type="checkbox" checked="">
							${_('言語で絞り込む (言語が取得可能なサービスのみ)')}
						</label>
					</li>
				</ul>
			</dd>

			<dt>${_('アラート音')}</dt>
			<dd>
				<button name="select-sound" type="button" aria-controls="alert-tone">${_('アラート音を選択')}</button>
				<button name="delete-sound" type="button" aria-controls="alert-tone" hidden="">
					${_('設定済みのアラート音を削除')}
				</button>
				<input hidden="" type="file" accept="audio/*" />
			</dd>

			<dt>${_('設定のインポートとエクスポート')}</dt>
			<dd>
				<button name="import" type="button">${_('JSONファイルからインポートする')}</button>
				<button name="export" type="button">${_('JSON形式でファイルにエクスポート')}</button>
			</dd>

			<dt></dt>
			<dd>
				<ul>
					<li>${_('項目名クリックで番組を昇順・降順に並べ替えることができます。')}</li>
					<li>${_('項目名をドラッグ&ドロップで列の位置を変更できます。')}</li>
					<li>${_('ユーザー名やコミュニティ名をテキストエリアにドラッグ&ドロップで除外 URL に指定できます。')}</li>
				</ul>
			</dd>
		</dl>
	`);

	/**
	 * 列IDのリスト。
	 * @type {string[]}
	 */
	const columnNames = Array.from(table.querySelectorAll('thead th')).map(th => th.id);

	/**
	 * 配信サービスのIDのリスト。
	 * @type {string[]}
	 */
	const serviceIds = Alert.services.map(service => service.id);

	/**
	 * ユーザー設定をインポートする際に利用するJSONスキーマ。
	 * @type {Object}
	 */
	UserSettings.schema = {
		type: 'object',
		properties: {
			version: {
				type: 'string',
			},
			words: {
				type: 'array',
				items: {
					type: 'string',
				},
			},
			NGs: {
				type: 'array',
				items: {
					type: 'string',
				},
			},
			NGsURI: {
				type: 'string',
			},
			order: {
				type: 'object',
				required: ['name', 'order'],
				properties: {
					name: {
						type: 'string',
						enum: columnNames,
					},
					order: {
						type: 'string',
						enum: ['asc', 'desc'],
					},
				},
				default: function () {
					const sortedTH = document.querySelector('#programs [data-sorted]');
					return {
						name: sortedTH.id,
						order: sortedTH.dataset.sorted,
					};
				}(),
			},
			'columns-position': {
				type: 'array',
				uniqueItems: true,
				items: {
					type: 'string',
					enum: columnNames,
				},
				default: columnNames,
			},
			'visible-columns': {
				type: 'array',
				uniqueItems: true,
				items: {
					type: 'string',
					enum: columnNames,
				},
				default: [
					'service', 'member_only', 'pubDate', 'title', 'category', 'owner_name', 'description',
					'view', 'num_res', 'community_name',
				],
			},
			'target-services': {
				type: 'array',
				uniqueItems: true,
				items: {
					type: 'string',
					enum: serviceIds.concat(['koebu-live', 'ustream', 'livetube']),
				},
				default: serviceIds.filter(
					serviceId => !['youtube-live', 'younow', 'livestream'].includes(serviceId)
				),
			},
			exclusionMemberOnly: {
				type: 'boolean',
				default: document.getElementsByName('exclusionMemberOnly')[0].checked,
			},
			ellipsisTooLongRSSData: {
				type: 'boolean',
				default: document.getElementsByName('ellipsisTooLongRSSData')[0].checked,
			},
			languageFilter: {
				type: 'boolean',
				default: document.getElementsByName('languageFilter')[0].checked,
			},
			audioMuted: {
				type: 'boolean',
				default: false,
			},
			audioVolume: {
				type: 'number',
				minimum: 0,
				maximum: 1,
				default: 1.0,
			},
			audioData: {
				type: 'string',
				pattern: '^data:(audio/[-_.0-9A-Za-z]+|video/ogg);base64,',
			},
		},
	};

	/**
	 * 設定保存時のバージョン番号。
	 * @type {?string}
	 */
	const version = await GM.getValue('version');

	// 検索語句
	let words = await GM.getValue('words');
	if (words) {
		words = JSON.parse(words);
		UserSettings.words = UserSettings.parseWords(words);
		document.getElementById('searching-words').value = words.join('\n') + '\n';
	}

	// 除外リスト
	let NGs = await GM.getValue('NGs');
	if (NGs) {
		NGs = JSON.parse(NGs);
		UserSettings.exclusions = UserSettings.parseExclusions(NGs);
		if (!version) {
			// 5.0.0 より前のバージョンの設定であれば
			await GM.setValue('NGs', JSON.stringify(UserSettings.exclusions));
		}
		document.getElementById('ng-communities').value = UserSettings.exclusions.join('\n') + '\n';
	}

	// 検索対象のサービス
	const targetServicesJSON = await GM.getValue('target-services');
	if (targetServicesJSON) {
		UserSettings.enableServices(version, JSON.parse(targetServicesJSON));
	} else {
		UserSettings.enableServices(GM.info.script.version, UserSettings.schema.properties['target-services'].default);
	}

	// プライベート配信・長い文字列の省略・言語
	for (const key of ['exclusionMemberOnly', 'ellipsisTooLongRSSData', 'languageFilter']) {
		const input = document.getElementsByName(key)[0];
		const savedValue = await GM.getValue(key);
		if (savedValue !== undefined && savedValue !== null) {
			input.checked = savedValue;
		}
	}

	/**
	 * アラート音を鳴らすための要素。
	 * @type {HTMLAudioElement}
	 */
	const alertTone = document.getElementById('alert-tone');

	// 音声ファイル
	const audioData = await UserSettings.getLargeValue('audioData');
	if (audioData) {
		alertTone.src = audioData;
		alertTone.hidden = false;
		document.getElementsByName('delete-sound')[0].hidden = false;
	}

	// ミュート
	if (await GM.getValue('audioMuted')) {
		alertTone.muted = true;
	}

	// 音量
	const audioVolume = await GM.getValue('audioVolume');
	if (audioVolume !== undefined && audioVolume !== null) {
		alertTone.volume = audioVolume;
	}

	// volumechangeイベント
	alertTone.addEventListener('volumechange', function (event) {
		if (event.target.muted) {
			GM.setValue('audioMuted', true);
		} else {
			GM.deleteValue('audioMuted');
		}

		if (event.target.volume === UserSettings.schema.properties.audioVolume.default) {
			GM.deleteValue('audioVolume');
		} else {
			GM.setValue('audioVolume', event.target.volume.toString());
		}
	});

	// clickイベント
	main.addEventListener('click', async function (event) {
		const target = event.target;
		const name = target.name;
		if (target.localName === 'button') {
			if (name === 'sets-blacklist-uri') {
				// 特定の URL から NG リストを読み込んで、検索時に付加
				const newURL = window.prompt(_('特定の URL から、除外 URL のリストを読み込み、検索時に付加します。') + '\n'
					+ _('JSON 形式の URL 文字列の配列のみ有効です。') + '\n'
					+ _('また、除外 URL リストの読み込みは、アラートページ読み込み時に1回だけ行われます。'), await GM.getValue('NGsURI', ''));
				if (newURL !== null) {
					GM.setValue('NGsURI', newURL);
				}
			} else if (name === 'import') {
				// インポート
				UserSettings.import();
			} else if (name === 'export') {
				// エクスポート
				UserSettings.export();
			} else if (name === 'select-sound') {
				// アラート音の選択
				document.querySelector('[accept="audio/*"]').click();
			} else if (name === 'delete-sound') {
				// 設定済みのアラート音の削除
				target.hidden = true;
				document.getElementById('alert-tone').hidden = true;
				document.querySelector('[accept="audio/*"]').src = '';
				UserSettings.deleteLargeValue('audioData');
			} else if (target.getAttribute('aria-controls') === 'user-setting-options') {
				// 追加設定ボックスの開閉
				const previousOpened = target.getAttribute('aria-expanded') === 'true';
				target.setAttribute('aria-expanded', previousOpened ? 'false' : 'true');
				document.getElementById('user-setting-options')
					.setAttribute('aria-hidden', previousOpened ? 'true' : 'false');
				target.scrollIntoView(!previousOpened);
				if (!previousOpened && !document.body.classList.contains('nofix')) {
					window.scrollBy(0, -document.getElementById('siteHeader').clientHeight);
				}
			} else if (name === 'save-searching-words' || name === 'save-ng-communities') {
				// 検索語句、検索から除外するユーザー・コミュニティ・チャンネルの保存
				// 二重クリックを防止
				target.disabled = true;

				/**
				 * 対応するテキストエリア。
				 * @type {string}
				 */
				const textarea = document.getElementById(target.getAttribute('aria-controls'));

				/**
				 * 前後の空白を削除したテキストエリアの値。
				 * @type {string}
				 */
				const trimedValue = textarea.value.trim();

				/**
				 * 正規化後、空行を含めずに改行で分割し、重複を削除した値。
				 * @type {string[]}
				 */
				const values = trimedValue === ''
					? []
					: Array.from(new Set(trimedValue.split(/\s*\n\s*/).map(Normalizer.normalize)));

				/**
				 * 検索語句の保存なら真。
				 * @type {boolean}
				 */
				const seavingWords = name === 'save-searching-words';

				/**
				 * {@link GM.setValue} に保存する値。
				 * @type {string[]}
				 */
				let savedValues;

				// 解析、キャッシュ、保存
				if (values.length > 0) {
					if (seavingWords) {
						// 検索語句
						savedValues = values;
						UserSettings.words = UserSettings.parseWords(values);
					} else {
						// 検索から除外するユーザー・コミュニティ・チャンネル
						savedValues = UserSettings.parseExclusions(values);
						UserSettings.exclusions = savedValues.concat(UserSettings.exclusionsFromExternal);
						// すでに表示している番組を非表示に
						TableProcessor.removeExclusions();
					}
				} else {
					savedValues = [];
				}

				/**
				 * テキストエリアに出力しなおす、正規化後の入力値。末尾に空行を含みます。
				 * @type {string}
				 */
				let outputValue;

				// 保存
				const gmValueName = seavingWords ? 'words' : 'NGs';
				if (savedValues.length > 0) {
					await GM.setValue(gmValueName, JSON.stringify(savedValues));
					outputValue = savedValues.join('\n') + '\n';
				} else {
					await GM.deleteValue(gmValueName);
					outputValue = '';
				}

				// テキストエリアに正規化後の文字列を出力
				if (textarea.value !== outputValue) {
					textarea.value = outputValue;
				}

				if (seavingWords) {
					Alert.restart();
				}

				// クリック禁止を解除
				target.disabled = false;
			}
		} else if (target.localName === 'th') {
			// 列ごとの並び替え
			TableProcessor.sort(target);
			event.target.blur();
		}
	});

	// changeイベント
	document.getElementById('user-setting-options').addEventListener('change', function (event) {
		const input = event.target;

		switch (input.name) {
			case 'visible-columns': {
				// 表示する項目の選択
				const th = document.getElementById(input.value);
				if (input.checked) {
					TableProcessor.showColumn(th);
				} else {
					TableProcessor.hideColumn(th);
				}
				GM.setValue('visible-columns', JSON.stringify(UserSettings.getShownColumns()));
				break;
			}

			case 'target-services':
				// 検索対象のサービスの選択
				if (input.checked) {
					Alert.enableService(input.value);
				} else {
					Alert.disableService(input.value);
				}
				GM.setValue('target-services', JSON.stringify(UserSettings.getTargetServices()));
				break;

			case 'exclusionMemberOnly':
				// プライベート配信の非表示
				if (input.checked) {
					TableProcessor.removePrivatePrograms();
					GM.setValue('exclusionMemberOnly', true);
				} else {
					GM.deleteValue('exclusionMemberOnly');
				}
				break;

			case 'ellipsisTooLongRSSData':
				// 文字数制限を超えている場合に省略
				if (input.checked) {
					GM.deleteValue('ellipsisTooLongRSSData');
				} else {
					GM.setValue('ellipsisTooLongRSSData', false);
				}
				break;

			case 'languageFilter':
				// 言語で絞り込み
				if (input.checked) {
					GM.deleteValue('languageFilter');
				} else {
					GM.setValue('languageFilter', false);
				}
				break;

			default:
				if (input.accept === 'audio/*') {
					// 音楽ファイルが選択された時
					const file = input.files[0];
					if (file) {
						const alertTone = document.getElementById('alert-tone');
						if (alertTone.canPlayType(file.type)) {
							const alertReader = new FileReader();
							alertReader.addEventListener('load', function (event) {
								UserSettings.setAudioData(event.target.result);
							});
							alertReader.readAsDataURL(file);
						} else {
							window.alert(_('使用中のブラウザが対応していないファイル形式です。'));
						}
					}
				}
		}
	});

	// ダブルクリックした検索語句・除外URLを新しいタブで開く
	document.getElementById('main-settings').addEventListener('dblclick', function (event) {
		const textarea = event.target;
		if (textarea.localName === 'textarea') {
			/**
			 * ダブルクリックされた位置。
			 * @type {number}
			 */
			const clickedPosition = textarea.selectionStart;

			// ダブルクリックされた行を取得
			const words = textarea.value;
			let beginSlice = words.lastIndexOf(
				'\n',
				words.slice(clickedPosition, clickedPosition + 1) === '\n' ? clickedPosition - 1 : clickedPosition
			);
			if (beginSlice === -1) {
				beginSlice = 0;
			}
			const endSlice = words.indexOf('\n', clickedPosition);
			const word = words.slice(beginSlice === -1 ? 0 : beginSlice, endSlice === -1 ? undefined : endSlice).trim();

			if (word) {
				const url = textarea.name === 'searching-words'
					? 'http://live.nicovideo.jp/search/' + encodeURIComponent(word)
					: UserSettings.parseExclusion(word);
				if (url) {
					window.open(url);
				}
			}
		}
	});

	/**
	 * 現在のドラッグイベント。
	 * @type {DragEvent}
	 */
	let draggingEvent;

	// 列の位置
	table.addEventListener('dragstart', function (event) {
		draggingEvent = event;
		if (event.target.localName === 'th' && document.querySelector('#programs thead tr').contains(event.target)) {
			event.dataTransfer.setData('Text', '');
			// 他のスクリプトを抑制
			event.stopPropagation();
		}
	});

	const headRow = document.querySelector('#programs thead tr');
	headRow.addEventListener('drag', function (event) {
		if (event.target.localName === 'th') {
			// 他のスクリプトを抑制
			event.stopPropagation();
		}
	});

	headRow.addEventListener('dragover', function (event) {
		if (draggingEvent.target.localName === 'th' && event.currentTarget.contains(draggingEvent.target)) {
			event.preventDefault();
			// 項目の移動先
			const className = event.pageX < event.target.offsetLeft + event.target.offsetWidth / 2
				? 'inserting-before'
				: 'inserting-after';
			if (!event.target.classList.contains(className)) {
				TableProcessor.removeOldClassName();
				event.target.classList.add(className);
			}
		}
	});

	headRow.addEventListener('dragleave', function (event) {
		if (draggingEvent.target.localName === 'th' && event.currentTarget.contains(draggingEvent.target)) {
			TableProcessor.removeOldClassName();
		}
	});

	main.addEventListener('drop', async function (event) {
		const row = event.currentTarget.querySelector('thead tr');
		if (draggingEvent.target.localName === 'th' && row.contains(draggingEvent.target)
			&& event.target.localName === 'th' && row.contains(event.target)) {
			// 項目を移動
			const refIndex = event.target.cellIndex + (event.target.classList.contains('inserting-before') ? 0 : 1);
			if (draggingEvent.target.cellIndex !== refIndex) {
				// 変更があれば
				TableProcessor.moveColumn(draggingEvent.target, refIndex);

				// 設定を保存
				await GM.setValue('columns-position', JSON.stringify(TableProcessor.getColumnPositions()));
			}

			event.target.classList.remove('inserting-before');
			event.target.classList.remove('inserting-after');
		} else if (event.target.name === 'ng-communities') {
			// NG コミュニティを追加
			event.preventDefault();

			// 他のスクリプトを阻害しないよう dragend イベントを発生させておく
			const init = {};
			for (const key in draggingEvent) {
				init[key] = draggingEvent[key];
			}
			draggingEvent.target.dispatchEvent(new DragEvent('dragend', init));

			event.target.value += '\n' + event.dataTransfer.getData('Text');
			document.getElementsByName('save-ng-communities')[0].click();
		}
	});

	// 列の位置
	const columnPositions = await GM.getValue('columns-position');
	if (columnPositions) {
		await TableProcessor.reflectColumnPositions(version, JSON.parse(columnPositions));
	}

	// 並び順
	let order = await GM.getValue('order');
	if (order) {
		order = JSON.parse(order);
		if (order.name !== UserSettings.schema.properties.order.default.name
			|| order.order !== UserSettings.schema.properties.order.default.order) {
			TableProcessor.sort(document.getElementById(order.name));
		} else {
			await GM.deleteValue('order');
		}
	}

	// 表示される列
	const visibleColumnsJSON = await GM.getValue('visible-columns');
	await UserSettings.showColumns(version, visibleColumnsJSON
		? JSON.parse(visibleColumnsJSON)
		: UserSettings.schema.properties['visible-columns'].default);

	// 現在のバージョン番号を保存
	await GM.setValue('version', GM.info.script.version);

	// 外部からの除外リスト取得
	const NGsURI = await GM.getValue('NGsURI');
	let preprocessing;
	if (NGsURI) {
		preprocessing = new HTTPRequest({
			method: 'GET',
			url: NGsURI,
			responseType: 'json',
			timeout: 30 * DateUtils.SECONDS_TO_MILISECONDS,
			mode: 'no-cors',
		}).send().then(function (response) {
			UserSettings.exclusionsFromExternal = UserSettings.parseExclusions(response);
			UserSettings.exclusions = UserSettings.exclusions.concat(UserSettings.exclusionsFromExternal);
		}).catch(function (error) {
			window.alert(_('指定された URL から、除外 URL リストを読み込めませんでした。\n取得せずに続行します。\n\nエラーメッセージ:\n%s').replace('%s', error));
		}).then();
	} else {
		preprocessing = Promise.resolve();
	}

	// 検索開始
	preprocessing.then(function () {
		Alert.initialize();
		Alert.restart();

		// 経過時間の定期的な更新
		TableProcessor.startUpdatingDurations();
	});
}

})();