ニコ生アラート(簡)

Alerts you to live streams that match your search. Supports these sites: FC2 Live, CaveTube, koebu LIVE!, SHOWROOM, Stickam JAPAN!, TwitCasting, Twitch, Niconico Live, Himawari Stream, Livetube

2015-10-03 일자. 최신 버전을 확인하세요.

질문, 리뷰하거나, 이 스크립트를 신고하세요.
// ==UserScript==
// @name        ニコ生アラート(簡)
// @name:ja     ニコ生アラート (簡)
// @name:en     Nico Live Alert (Kan)
// @namespace   http://userscripts.org/users/347021
// @id          niconico-alert-keyword-347021
// @version     5.0.0
// @description Alerts you to live streams that match your search. Supports these sites:  FC2 Live, CaveTube, koebu LIVE!, SHOWROOM, Stickam JAPAN!, TwitCasting, Twitch, Niconico Live, Himawari Stream, Livetube
// @description:ja キーワードにヒットしたライブ配信を通知します。次のサイトに対応: FC2ライブ、CaveTube、 こえ部LIVE!、SHOWROOM、Stickam JAPAN!、ツイキャス、Twitch、ニコニコ生放送、ひまわりストリーム、Livetube
// @match       http://*.nicovideo.jp/*
// @match       *://live.fc2.com/*
// @match       http://gae.cavelis.net/*
// @match       http://koebu.com/*
// @match       https://www.showroom-live.com/*
// @match       http://www.stickam.jp/*
// @match       http://twitcasting.tv/*
// @match       http://www.twitch.tv/*
// @match       http://himast.in/*
// @match       *://www.ustream.tv/*
// @match       *://www.youtube.com/*
// @match       https://www.younow.com/*
// @match       *://livestream.com/*
// @match       *://livetube.cc/*
// @run-at      document-start
// @grant       GM_getValue
// @grant       GM_setValue
// @grant       GM_deleteValue
// @grant       GM_xmlhttpRequest
// @grant       GM_registerMenuCommand
// @grant       GM_openInTab
// @grant       GM_info
// @icon        
// @compatible  Firefox
// @compatible  Opera
// @compatible  Chrome
// @author      100の人
// @homepage    https://greasyfork.org/scripts/272
// @contributor HADAA
// @license     Mozilla Public License Version 2.0 (MPL 2.0); https://www.mozilla.org/MPL/2.0/
// ==/UserScript==
	
// For UserScriptLoader.uc.js
if (typeof GM_info === 'undefined') {
	window.GM_info = {
		script: {
			version: '5.0.0',
		},
	};
}

(function () {
'use strict';

defineGettext();

// L10N
Gettext.setLocalizedTexts({
	'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',
		'こえ部LIVE!': 'koebu LIVE!',
		'SHOWROOM': 'SHOWROOM',
		'Stickam JAPAN!': 'Stickam JAPAN!',
		'ツイキャス': 'TwitCasting',
		'Twitch': 'Twitch',
		'ニコニコ生放送': 'Niconico Live',
		'ひまわりストリーム': 'Himawari Stream',
		'Ustream': 'Ustream',
		'Youtube ライブ': 'Youtube Live',
		'YouNow': 'YouNow',
		'Livestream': 'Livestream',
		'Livetube': 'Livetube',
		'表示する項目の設定': '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',
		'設定済みのアラート音を削除': 'Deletes alert sound set in advance',
		' ❰❰%s❱❱ ': ' <<%s>> ', // ツールチップ内における一致箇所のマーク
		
		'ニコ生アラート (簡)': 'Nico Live alert (Kan)',
	},
});

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

/**
 * メインの処理を行うユーティリティークラス。
 */
var Alert = {
	/**
	 * ページタイトル、ブラウジングコンテキスト名、ユーザースクリプトのコマンド名に用いる文字列。
	 * @constant {string}
	 */
	NAME: _('ニコ生アラート (簡)'),
	
	/**
	 * {@link GM_setLargeString} や URL に用いる文字列。
	 * @constant {string}
	 */
	ID: 'alert-keyword-347021',
}

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

if (window.location.href !== alertPageURL) {
	GM_registerMenuCommand(Alert.NAME, function () {
		GM_openInTab(alertPageURL);
	});
} else {
	polyfill();
	startScript(
		main,
		parent => parent.localName === 'body',
		target => target.id === 'utility_link',
		() => document.getElementById('utility_link')
	);
}

function main () {
	defineClasses();
	
	// ページタイトル
	document.title = Alert.NAME;

	// Favicon
	var icon = document.querySelector('[rel="icon"]');
	icon.href = Alert.ICON;
	document.head.appendChild(icon);
	
	// 元のページ内容を削除
	document.getElementById('all_cover').remove();

	// ヘッダを修正
	for (var select of document.querySelectorAll('[href^="javascript:"]')) {
		select.target = '_self';
	}
	
	// リンク先を新しいタブで開く、スタイルの設定
	document.head.insertAdjacentHTML('beforeend', h`
		<base target="_blank" />
		
		<style>
			[aria-hidden="true"] {
				display: none;
			}
			
			main {
				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 {
				-webkit-column-count: 2;    /* Opera and Google Chrome */
				-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" sortable="">
			<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="${_('配信開始からの経過時間')}" sorted="" 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}
	 */
	var table = document.getElementById('programs');
	
	if (!('sortable' in table)) {
		document.head.insertAdjacentHTML('beforeend', `<style>
			/*====================================
				列ごとの並べ替え
			 */
			#programs thead th:hover,
			#programs thead th:focus {
				cursor: pointer;
				background: gainsboro;
			}
			
			/*------------------------------------
				▲▼
			 */
			[sorted]::before {
				content: "▲";
				color: darkslategray;
				margin-right: 0.5em;
			}
			[sorted*="reversed"]::before {
				content: "▼";
			}
		</style>`);
		
		for (var 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}
	 */
	var 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[]}
	 */
	var columnNames = Array.from(table.querySelectorAll('thead th')).map(th => th.id);
	
	/**
	 * 配信サービスのIDのリスト。
	 * @type {string[]}
	 */
	var 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 () {
					var sortedTH = document.querySelector('#programs [sorted]');
					return {
						name: sortedTH.id,
						order: TableProcessor.getSorting(sortedTH),
					};
				}(),
			},
			'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,
				},
				default: serviceIds.filter(serviceId => ['ustream', 'youtube-live', 'younow', 'livestream'].indexOf(serviceId) === -1),
			},
			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}
	 */
	var version = GM_getValue('version');
	
	// 検索語句
	var words = GM_getValue('words');
	if (words) {
		words = JSON.parse(words);
		UserSettings.words = UserSettings.parseWords(words);
		document.getElementById('searching-words').value = words.join('\n') + '\n';
	}
	
	// 除外リスト
	var NGs = GM_getValue('NGs');
	if (NGs) {
		NGs = JSON.parse(NGs);
		UserSettings.exclusions = UserSettings.parseExclusions(NGs);
		if (!version) {
			// 5.0.0 より前のバージョンの設定であれば
			GM_setValue('NGs', JSON.stringify(UserSettings.exclusions));
		}
		document.getElementById('ng-communities').value = UserSettings.exclusions.join('\n') + '\n';
	}
	
	// 検索対象のサービス
	var targetServicesJSON = 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 (var key of ['exclusionMemberOnly', 'ellipsisTooLongRSSData', 'languageFilter']) {
		var input = document.getElementsByName(key)[0];
		var savedValue = GM_getValue(key);
		if (savedValue !== undefined && savedValue !== null) {
			input.checked = savedValue;
		}
	}
	
	/**
	 * アラート音を鳴らすための要素。
	 * @type {HTMLAudioElement}
	 */
	var alertTone = document.getElementById('alert-tone');
	
	// 音声ファイル
	var audioData = UserSettings.getLargeValue('audioData');
	if (audioData) {
		alertTone.src = audioData;
		alertTone.hidden = false;
		document.getElementsByName('delete-sound')[0].hidden = false;
	}
	
	// ミュート
	if (GM_getValue('audioMuted')) {
		alertTone.muted = true;
	}
	
	// 音量
	var audioVolume = 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', function (event) {
		var target = event.target;
		var name = target.name;
		if (target.localName === 'button') {
			if (name === 'sets-blacklist-uri') {
				// 特定の URL から NG リストを読み込んで、検索時に付加
				var newURL = window.prompt(_('特定の URL から、除外 URL のリストを読み込み、検索時に付加します。') + '\n'
					+ _('JSON 形式の URL 文字列の配列のみ有効です。') + '\n'
					+ _('また、除外 URL リストの読み込みは、アラートページ読み込み時に1回だけ行われます。'), 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') {
				// 追加設定ボックスの開閉
				var 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}
				 */
				var textarea = document.getElementById(target.getAttribute('aria-controls'));
	
				/**
				 * 前後の空白を削除したテキストエリアの値。
				 * @type {string}
				 */
				var trimedValue = textarea.value.trim();
				
				/**
				 * 正規化後、空行を含めずに改行で分割し、重複を削除した値。
				 * @type {string[]}
				 */
				var values = trimedValue === ''
					? []
					: Array.from(new Set(trimedValue.split(/\s*\n\s*/).map(StringProcessor.normalize)));
				
				/**
				 * 検索語句の保存なら真。
				 * @type {boolean}
				 */
				var seavingWords = name === 'save-searching-words';
				
				/**
				 * {@link GM_setValue} に保存する値。
				 * @type {string[]}
				 */
				var 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}
				 */
				var outputValue;
				
				// 保存
				var gmValueName = seavingWords ? 'words' : 'NGs';
				if (savedValues.length > 0) {
					GM_setValue(gmValueName, JSON.stringify(savedValues));
					outputValue = savedValues.join('\n') + '\n';
				} else {
					GM_deleteValue(gmValueName);
					outputValue = '';
				}
	
				// テキストエリアに正規化後の文字列を出力
				if (textarea.value !== outputValue) {
					textarea.value = outputValue;
				}
				
				if (seavingWords) {
					Alert.restart();
				}
				
				// クリック禁止を解除
				target.disabled = false;
			}
		} else if (target.localName === 'th') {
			// 列ごとの並び替え
			if (!('sorted' in target)) {
				TableProcessor.sort(target);
				event.target.blur();
			}
		}
	});
	
	// changeイベント
	document.getElementById('user-setting-options').addEventListener('change', function (event) {
		var input = event.target;
		
		switch (input.name) {
			case 'visible-columns':
				// 表示する項目の選択
				var 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/*') {
					// 音楽ファイルが選択された時
					var file = input.files[0];
					if (file) {
						var alertTone = document.getElementById('alert-tone');
						if (alertTone.canPlayType(file.type)) {
							var 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) {
		var textarea = event.target;
		if (textarea.localName === 'textarea') {
			/**
			 * ダブルクリックされた位置。
			 * @type {number}
			 */
			var clickedPosition = textarea.selectionStart;
		
			// ダブルクリックされた行を取得
			var words = textarea.value;
			var beginSlice = words.lastIndexOf(
				'\n',
				words.slice(clickedPosition, clickedPosition + 1) === '\n' ? clickedPosition - 1 : clickedPosition
			);
			if (beginSlice === -1) {
				beginSlice = 0;
			}
			var endSlice = words.indexOf('\n', clickedPosition);
			var word = words.slice(beginSlice === -1 ? 0 : beginSlice, endSlice === -1 ? undefined : endSlice).trim();
		
			if (word) {
				var url = textarea.name === 'searching-words'
					? 'http://live.nicovideo.jp/search/' + encodeURIComponent(word)
					: UserSettings.parseExclusion(word);
				if (url) {
					window.open(url);
				}
			}
		}
	});
	
	/**
	 * 現在のドラッグイベント。
	 * @type {DragEvent}
	 */
	var 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();
		}
	});
	
	var 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();
			// 項目の移動先
			var 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', function (event) {
		var row = event.currentTarget.querySelector('thead tr');
		if (draggingEvent.target.localName === 'th' && row.contains(draggingEvent.target) && event.target.localName === 'th' && row.contains(event.target)) {
			// 項目を移動
			var refIndex = event.target.cellIndex + (event.target.classList.contains('inserting-before') ? 0 : 1);
			if (draggingEvent.target.cellIndex !== refIndex) {
				// 変更があれば
				TableProcessor.moveColumn(draggingEvent.target, refIndex);
	
				// 設定を保存
				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 イベントを発生させておく
			var init = {};
			for (var 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();
		}
	});
	
	// 列の位置
	var columnPositions = GM_getValue('columns-position');
	if (columnPositions) {
		TableProcessor.reflectColumnPositions(version, JSON.parse(columnPositions));
	}
	
	// 並び順
	var order = 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 {
			GM_deleteValue('order')
		}
	}
	
	// ソート時に並び順を保存
	table.addEventListener('sort', function (event) {
		var tHead = event.target.tHead;
		new MutationObserver(function (mutations, observer) {
			observer.disconnect();
			var sortedTH = tHead.querySelector('[sorted]');
			GM_setValue('order', JSON.stringify({
				name: sortedTH.id,
				order: TableProcessor.getSorting(sortedTH),
			}));
		}).observe(tHead, {
			subtree: true,
			attributeFilter: ['sorted'],
		});
	});
	
	// 表示される列
	var visibleColumnsJSON = GM_getValue('visible-columns');
	UserSettings.showColumns(version, visibleColumnsJSON
		? JSON.parse(visibleColumnsJSON)
		: UserSettings.schema.properties['visible-columns'].default);
	
	// 現在のバージョン番号を保存
	GM_setValue('version', GM_info.script.version);

	// 外部からの除外リスト取得
	var NGsURI = GM_getValue('NGsURI');
	var 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();
	});
}

function defineClasses() {
	/**
	 * ページのFaviconに用いるデータURL。
	 * @constant {string}
	 */
	Alert.ICON = '';
	
	/**
	 * 初期化。
	 */
	Alert.initialize = function () {
		// 各サービスのインスタンスにイベントリスナーを設定
		for (var 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',
		});
	};
	
	/**
	 * @param {Service#ProgramEvent}
	 * @access private
	 */
	Alert.onprogress = function (event) {
		var program = event.detail;
		if (!program.private || !document.getElementsByName('exclusionMemberOnly')[0].checked) {
			// プライベート配信で無い、またはプライベート配信を拒否していなければ
			var 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);
				
				// アラート音を鳴らす
				var alertTone = document.getElementById('alert-tone');
				if (!alertTone.hidden && !alertTone.muted && alertTone.volume > 0) {
					alertTone.play();
				}
			}
		}
	};
	
	/**
	 * @param {Service#LoadedEvent}
	 * @access private
	 */
	Alert.onload = function (event) {
		// 配信終了の番組を削除
		TableProcessor.removeOldPrograms(event.detail.service, event.detail.programs, event.detail.searchCriteria);
		
		// 最終更新日時を設定
		UserSettings.showLatestUpdatedDate(event.target);
		
		// ヒット数の更新
		this.showHits();
	};
	
	/**
	 * ヒット数を更新します。
	 */
	Alert.showHits = function () {
		var tBody = document.querySelector('#programs tbody');
		
		/**
		 * ヒット数。
		 * @type {number}
		 */
		var 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');
		}
	};
	
	/**
	 * スクリプトが動作している状態であれば真。
	 * @type {boolean}
	 * @access private
	 */
	Alert.run = false;
	
	/**
	 * スクリプトが停止している状態であれば起動し、動作している状態であればOR検索ができないサービスを再読み込みします。
	 * 検索語句が空の場合は、スクリプトを停止します。
	 */
	Alert.restart = function () {
		if (UserSettings.words.length === 0) {
			// 検索語句が空であれば
			this.stop();
		} else if (this.run) {
			// スクリプトが動作している状態であれば
			for (var service of this.services) {
				if (service.disabledOr) {
					service.reset();
				}
			}
		} else {
			// スクリプトが停止している状態であれば
			// 有効なサービスで検索を開始
			this.run = true;
			var enabledServices = UserSettings.getTargetServices();
			for (var service of this.services) {
				if (enabledServices.indexOf(service.id) !== -1) {
					service.start();
				}
			}
		}
	};
	
	/**
	 * スクリプトを停止します。
	 */
	Alert.stop = function () {
		if (this.run) {
			// スクリプトが動作している状態であれば、すべてのサービスで検索を停止
			this.run = false;
			for (var service of this.services) {
				service.stop();
			}
		}
	};
	
	/**
	 * 指定されたサービスを有効化します。
	 * @param {string} id
	 */
	Alert.enableService = function (id) {
		if (this.run) {
			this.services.find(service => service.id === id).start();
		}
	},
	
	/**
	 * 指定されたサービスを無効化します。
	 * @param {string} id
	 */
	Alert.disableService = function (id) {
		if (this.run) {
			var service = this.services.find(service => service.id === id);
			service.stop();
			TableProcessor.removeProgramsWithService(service);
		}
	},
	
	/**
	 * ライブストリーミング配信サービスのリスト。
	 * @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) {
			var 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 : [['雑談', 'ゲーム', '作業', '動画', 'その他', , '公式'][program.category - 1]],
				author: {
					name: program.name,
					url: url,
				},
				visitors: Number.parseInt(program.total),
				language: program.lang,
			});
		},
	}), new Service({
		id: 'cavetube',
		name: _('CaveTube'),
		url: 'http://gae.cavelis.net/',
		disabledSearch: true,
		getDetails: function () {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'http://rss.cavelis.net/index_live.xml',
				responseType: 'document',
			};
		},
		parseResponse: function (response) {
			return { programs: response.getElementsByTagNameNS(DOMUtils.ATOM_NAMESPACE, 'entry') };
		},
		/**
		 * CaveTube名前空間。
		 * @constant {string}
		 */
		CT_NAMESPACE: 'http://gae.cavelis.net',
		convertIntoEntry: function (program) {
			var name = program.getElementsByTagNameNS(DOMUtils.ATOM_NAMESPACE, 'name')[0].textContent;
			var parser = new DOMParser();
			return new Program(
				program.getElementsByTagNameNS(DOMUtils.ATOM_NAMESPACE, 'link')[0].getAttribute('href'),
				parser.parseFromString(program.getElementsByTagNameNS(DOMUtils.ATOM_NAMESPACE, 'title')[0].textContent, 'text/html').body.textContent,
				{
					icon: new URL(program.getElementsByTagNameNS(this.CT_NAMESPACE, 'thumbnail_path')[0].textContent, 'http://gae.cavelis.net/').href,
					published: new Date(program.getElementsByTagNameNS(DOMUtils.ATOM_NAMESPACE, 'published')[0].textContent),
					author: {
						name: name,
						url: 'http://gae.cavelis.net/user/' + encodeURIComponent(name),
					},
					categories: program.getElementsByTagNameNS(this.CT_NAMESPACE, 'tag')[0].textContent.split(' '),
					summary: parser.parseFromString(program.getElementsByTagNameNS(DOMUtils.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: 'koebu-live',
		name: _('こえ部LIVE!'),
		url: 'http://koebu.com/live/',
		icon: 'http://koebu.com/favicon.ico',
		disabledSearch: true,
		getDetails: function () {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'http://koebu.com/live/list?page=1',
				responseType: 'document',
			};
		},
		parseResponse: function (response, details) {
			var programs = { programs: response.getElementsByClassName('unitCh') };
			var nextPage = response.querySelector('.pager .next a');
			if (nextPage) {
				details.url = nextPage.href;
				programs.next = details;
			}
			return programs;
		},
		convertIntoEntry: function (program) {
			var unitThumb = program.querySelector('.unitThumb img');
			var masterUnitChThumb = program.querySelector('.masterUnitCh img');
			var link = unitThumb.parentElement.href;
			return new Program(link, unitThumb.alt, {
				icon: unitThumb.src,
				author: {
					name: masterUnitChThumb.alt,
					url: masterUnitChThumb.parentElement.href,
				},
				categories: Array.from(program.querySelectorAll('[rel="tag"]'), function (anchor) {
					return anchor.text;
				}),
				summary: program.getElementsByClassName('description')[0].textContent,
				visitors: Number.parseInt(program.querySelector('.ttlNumListen ~ dd').textContent),
				community: {
					name: unitThumb.alt,
					url: link,
				},
			});
		},
	}), 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: 'http://www.stickam.jp/',
		disabledSearch: true,
		getDetails: function () {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'http://www.stickam.jp/explore/session?page=1',
				responseType: 'document',
			};
		},
		parseResponse: function (response, details) {
			var programs = { programs: response.getElementsByClassName('col-md-3 col-sm-3') };
			var 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) {
			var userNameLink = program.getElementsByTagName('a')[0];
			return new Program(userNameLink.href, userNameLink.text, {
				icon: program.getElementsByClassName('embed-responsive-item')[0].dataset.src,
				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: 'http://twitcasting.tv/',
		disabledOr: true,
		disabledMinus: true,
		disabledLanguageFilter: true,
		getDetails: function (searchCriteria) {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'http://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) {
			var url = program.querySelector('.searcheduser a').href;
			var titleAndComments = /(.*?)(?: \((0|[1-9][0-9]*)\))?$/.exec(program.querySelector('.title a').text);
			var language;
			switch (program.getElementsByClassName('countryflag')[0].src) {
				case 'http://twitcasting.tv/img/c/us.gif':
					language = 'en';
					break;
				case 'http://twitcasting.tv/img/c/br.gif':
					language = 'pt';
					break;
				case 'http://twitcasting.tv/img/c/mx.gif':
					language = 'es';
					break;
				case 'http://twitcasting.tv/img/c/jp.gif':
					language = 'ja';
					break;
			}
			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: 'http://www.twitch.tv/',
		/**
		 * 検索結果の最大件数。
		 * @constant {number}
		 */
		MAX_RESULT_LENGTH: 100,
		disabledOr: true,
		disabledMinus: true,
		disabledLanguageFilter: true,
		getDetails: function (searchCriteria) {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: `https://api.twitch.tv/kraken/search/streams?limit=${this.MAX_RESULT_LENGTH}&q=${encodeURIComponent(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({
		id: 'niconico-live',
		name: _('ニコニコ生放送'),
		url: 'http://live.nicovideo.jp/',
		icon: 'http://nl.simg.jp/public/inc/assets/zero/img/base/favicon.ico',
		/**
		 * 検索結果の最大件数。
		 * @constant {number}
		 */
		MAX_RESULT_LENGTH: 100,
		getDetails: function (words) {
			return {
				mode: 'cors',
				method: 'POST',
				url: 'http://api.search.nicovideo.jp/api/',
				responseType: 'text',
				data: {
					query: words.map(function (searchCriteria) {
						var word = searchCriteria.plus.join(' ');
						if (searchCriteria.minus.length > 0) {
							word += ' -' +  searchCriteria.minus.join(' -')
						}
						return word;
					}).join(' OR '),
					service: ['live'],
					search: ['title', 'tags', 'description'],
					join: [
						'cmsid',            // ID
						'title',            // タイトル
						'member_only',      // プライベート
						'start_time',       // 開始時刻
						'community_icon',   // コミュニティアイコン
						'tags',             // タグ
						'description',      // 詳細
						'community_id',     // コミュニティID
						'channel_id',
						'view_counter',
						'comment_counter',
					],
					filters: [
						{
							type: 'equal',
							field: 'live_status',
							value: 'onair',
						}
					],
					sort_by: 'start_time',
					from: 0,
					size: this.MAX_RESULT_LENGTH,
					issuer: 'ニコ生アラート(簡)',
					reason: 'ma10',
				},
			};
		},
		parseResponse: function (response, details) {
			var programs = { programs: [] };
			for (var jsonString of response.trim().split('\n')) {
				var data = JSON.parse(jsonString);
				if (data.type === 'hits') {
					if (data.endofstream) {
						break;
					}
					programs.programs = data.values;
					if (programs.programs.length === details.data.size) {
						details.data.from = programs.programs[details.data.size - 1]._rowid + 1;
						programs.next = details;
					}
					break;
				}
			}
			return programs;
		},
		convertIntoEntry: function (program) {
			var otherDetails = {
				icon: program.community_icon,
				private: Boolean(program.member_only),
				published: new Date(program.start_time.replace(' ', 'T') + '+09:00'),
				// <br /> タグを半角スペースに置き換え、他のタグは取り除く
				summary: program.description.replace(/<br( )\/>|<font[^>]+>|<\/?(?:font|b|i|s|u)>/g, '$1'),
				categories: program.tags.split(' '),
				visitors: program.view_counter,
				comments: program.comment_counter,
			};
			
			if (p(otherDetails.summary).includes('&')) {
				// 文字参照が含まれていれば
				otherDetails.summary = new DOMParser().parseFromString(otherDetails.summary, 'text/html').body.textContent;
			}
			
			if (program.community_id || program.channel_id) {
				otherDetails.community = {
					name: program.community_id ? 'co' + program.community_id : 'ch' + program.channel_id,
				};
				otherDetails.community.url = 'http://com.nicovideo.jp/community/' + otherDetails.community.name;
			}
			
			return new Program('http://live.nicovideo.jp/watch/' + program.cmsid, program.title, otherDetails);
		},
		delay: 1 * DateUtils.MINUTES_TO_MILISECONDS,
	}), new Service({
		id: 'himawari-stream',
		name: _('ひまわりストリーム'),
		url: 'http://himast.in/',
		disabledSearch: true,
		getDetails: function () {
			return {
				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) {
			var details = [];
			var description = new DOMParser().parseFromString(program.getElementsByTagName('description')[0].textContent, 'text/html');
			var summary;
			for (var 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: 'ustream',
		name: _('Ustream'),
		url: 'http://www.ustream.tv/',
		icon: 'http://static-cdn1.ustream.tv/images/favicon-black:1.ico',
		disabledOr: true,
		disabledMinus: true,
		getDetails: function (searchCriteria) {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'https://www.ustream.tv/ajax/search.json?type=live&q=' + encodeURIComponent(searchCriteria.plus.join(' ')),
				responseType: 'json',
			};
		},
		parseResponse: function (response) {
			var doc = new DOMParser().parseFromString(response.pageContent, 'text/html');
			doc.head.insertAdjacentHTML('beforeend', h`<base href="${this.url}" />`);
			return { programs: doc.getElementsByClassName('media-item') };
		},
		convertIntoEntry: function (program) {
			var mediaData = program.getElementsByClassName('media-data')[0];
			var title = mediaData.title;
			var url = mediaData.href;
			var viewers = program.querySelector('.item-viewers strong');
			return new Program(url, title, {
				icon: program.querySelector('.channel-tbn img').src,
				visitors: viewers && Number.parseInt(viewers.textContent),
				community: {
					name: title,
					url: url,
				},
			});
		},
	}), 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) {
			var 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?filters=live&search_sort=video_date_uploaded&search_query='
					+ encodeURIComponent(query),
				responseType: 'document',
			};
		},
		parseResponse: function (response) {
			return { programs: response.getElementsByClassName('yt-lockup-dismissable') };
		},
		convertIntoEntry: function (program) {
			var anchor = program.querySelector('.yt-lockup-title a');
			var userAnchor = program.querySelector('.yt-lockup-byline a');
			var description = program.getElementsByClassName('yt-lockup-description')[0];
			var 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({
		id: 'younow',
		name: _('YouNow'),
		url: 'https://www.younow.com/',
		disabledMinus: true,
		/**
		 * 検索結果の最大件数。
		 * @constant {number}
		 */
		MAX_RESULT_LENGTH: 100,
		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 => {
					return {
						indexName: 'people_search_live',
						params: `hitsPerPage=${this.MAX_RESULT_LENGTH}&advancedSyntax=1&query=${encodeURIComponent(searchCriteria.plus.join(' '))}`,
					};
				}) },
			};
		},
		parseResponse: function (response, details, words) {
			return { programs: Array.prototype.concat.apply([], response.results.map(function (result, index) {
				var programs = [];
				var searchCriteria = words[index];
				for (var program of result.hits) {
					if (program.tag === '') {
						break;
					}
					program.searchCriteria = searchCriteria;
					programs.push(program);
				}
				return programs;
			})) };
		},
		convertIntoEntry: function (program) {
			var 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/794776b/assets/favicon-80e0433a0ae645d507841ae46338238a.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 => {
						var query = searchCriteria.plus.join(' ');
						if (searchCriteria.minus.length > 0) {
							query += ' -' +  searchCriteria.minus.join(' -')
						}
						return {
							indexName: 'events',
							params: `hitsPerPage=${this.MAX_RESULT_LENGTH}&advancedSyntax=1&facets=*&facetFilters=%5B%22is_live%3A1%22%5D&query=${encodeURIComponent(query)}`,
						};
					}),
				},
			};
		},
		parseResponse: function (response, details, words) {
			return { programs: Array.prototype.concat.apply([], response.results.map(function (result, index) {
				var searchCriteria = words[index];
				for (var program of result.hits) {
					program.searchCriteria = searchCriteria;
				}
				return result.hits;
			})) };
		},
		convertIntoEntry: function (program) {
			var 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);
		},
	}), new Service({
		id: 'livetube',
		name: _('Livetube'),
		url: 'https://livetube.cc/',
		disabledSearch: true,
		getDetails: function () {
			return {
				mode: 'no-cors',
				method: 'GET',
				url: 'https://livetube.cc/index.live.json',
				responseType: 'json',
			};
		},
		parseResponse: function (response) {
			return { programs: response };
		},
		convertIntoEntry: function (program) {
			return new Program('https://livetube.cc/' + program.link, program.title, {
				published: new Date(program.created),
				author: {
					name: program.author,
					url: 'https://livetube.cc/hina0083/' + encodeURIComponent(program.author),
				},
				categories: program.tags,
				visitors: program.view,
				comments: program.comments,
			});
		},
	})];
	
	/**
	 * ライブ配信番組を表示する表に関する処理を行うユーティリティークラス。
	 */
	window.TableProcessor = {
		/**
		 * 経過時間を更新する間隔。ミリ秒数。
		 * @type {number}
		 */
		DURATION_UPDATING_INTERVAL: 20 * DateUtils.SECONDS_TO_MILISECONDS,
		
		/**
		 * 経過時間の更新を開始します。
		 */
		startUpdatingDurations: function () {
			for (var duration of document.querySelectorAll('[itemtype="http://schema.org/VideoObject"] [itemprop="duration"]:not([hidden])')) {
				var serialized = DateUtils.getDuration(new Date(duration.dataset.start));
				duration.value = serialized.dateTime;
				duration.textContent = serialized.text;
			}
			window.setTimeout(this.startUpdatingDurations.bind(this), this.DURATION_UPDATING_INTERVAL);
		},
		
		/**
		 * プライベート番組を表から取り除きます。
		 * @returns {HTMLRowElement[]}
		 */
		removePrivatePrograms: function () {
			for (var requiresSubscription of document.querySelectorAll('[itemtype="http://schema.org/VideoObject"] [itemprop="requiresSubscription"][value="true"]')) {
				requiresSubscription.closest('[itemscope]').remove();
			}
			Alert.showHits();
		},
		
		/**
		 * 指定されたサービスの番組を表から取り除きます。
		 * @param {Service} service
		 * @returns {HTMLRowElement[]}
		 */
		removeProgramsWithService: function (service) {
			for (var 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 (var 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) {
					var 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) {
			var urls = currentPrograms.map(program => program.link);
			for (var 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) {
			var table = document.getElementById('programs');
			var anchor = table.querySelector(urls.map(url => `[href="${CSS.escape(program.link)}"]`).join(','));
			var previousRow;
			if (anchor) {
				// すでに同じ番組、または同じユーザーの番組、同じコミュニティの番組が追加されていれば
				previousRow = anchor.closest('[itemtype="http://schema.org/VideoObject"]');
			}
			
			var row = this.convertProgramToRow(program, previousRow);
			var tBody = table.tBodies[0];
			tBody.insertBefore(row, tBody.rows[this.getInsertPosition(table, row)]);
		},
		
		/**
		 * 指定された番組を表すtr要素を返します。
		 * @param {Program} program
		 * @param {HTMLTableRowElement} [previousRow] - 指定されていれば、その行を更新します。
		 * @return {HTMLTableRowElement}
		 */
		convertProgramToRow: function (program, previousRow) {
			var row = previousRow || document.querySelector('#programs template').content.firstElementChild.cloneNode(true);
			
			// サービス
			if (!previousRow) {
				var provider = row.querySelector('[itemprop="provider"]');
				provider.querySelector('[itemprop="name"]').value = program.service.name;
				provider.querySelector('[itemprop="url"]').href = program.service.url;
				var 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);
				}
			}
			
			// アイコン
			var 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;
			}
			
			// コミュ限
			var 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 = '';
				}
			}
			
			// 経過時間
			var duration = row.querySelector('[itemprop="duration"]');
			if (program.published) {
				if (!previousRow || program.published.toISOString() !== duration.dataset.start) {
					var serialized = DateUtils.getDuration(program.published);
					duration.dateTime = serialized.dateTime;
					duration.textContent = serialized.text;
					duration.dataset.start = program.published.toISOString();
					duration.hidden = false;
				}
			}
			
			// タイトル
			var 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);
			}
			
			// タグ
			var keywords = row.querySelector('[itemprop="keywords"]');
			var 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);
			}
			
			// 配信者
			var author = row.querySelector('[itemprop="author"]');
			if (program.author) {
				var name = author.querySelector('[itemprop="name"]');
				var 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;
					}
				}
			}
			
			// 説明文
			var 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);
			}
			
			// 累計来場者数
			var userInteractionCount = row.querySelector('[itemprop="userInteractionCount"]');
			if (typeof program.visitors === 'number') {
				userInteractionCount.value = program.visitors;
				userInteractionCount.textContent = _('%d 人').replace('%d', program.visitors);
			}
			
			// コメント数
			var commentCount = row.querySelector('[itemprop="commentCount"]');
			if (typeof program.comments === 'number') {
				commentCount.value = program.comments;
				commentCount.textContent = _('%d コメ').replace('%d', program.comments);
			}
			
			// コミュニティ
			var productionCompany = row.querySelector('[itemprop="productionCompany"]');
			if (program.community) {
				productionCompany.querySelector('[itemprop="name"]').value = program.community.name;
				var 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 (var 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[][]}
			 */
			var offsets = searchCriteria ? WordProcessor.getMatches(str, searchCriteria) : [];
			
			/**
			 * 挿入するTextノード。
			 * @type {Text}
			 */
			var text = new Text(str);
			
			/**
			 * mark要素に内包する範囲。
			 * @type {Range[]}
			 */
			var ranges = WordProcessor.convertOffsetsToRanges(text, offsets);
			
			if (str.length > UserSettings.MAX_VISIBLE_CHARACTERS
				&& document.getElementsByName('ellipsisTooLongRSSData')[0].checked) {
				// 文字数が制限を超えており、表示文字数の制限が有効なら
				DOMTokenList.prototype.add.apply(
					target.classList,
					WordProcessor.extractTextNode(text, offsets, UserSettings.MAX_VISIBLE_CHARACTERS, UserSettings.MAX_BEFORE_CHARACTERS)
				);
				var title = WordProcessor.markMatchesAsPlainText(str, offsets);
				if (/\(Windows .+ Chrome\/.+(?!Edge\/)/.test(window.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.appendChild(text);
			for (var range of ranges) {
				range.surroundContents(document.createElement('mark'));
			}
		},
	
		/**
		 * 指定された列が昇順に並んでいれば「asc」、降順に並んでいれば「desc」を返します。
		 * @param {HTMLTableHeaderCellElement} sortedTH
		 */
		getSorting: function (sortedTH) {
			return sortedTH.getAttribute('sorted').toLowerCase().trim().split(/\s+/).indexOf('reversed') !== -1
				? 'desc'
				: 'asc';
		},
	
		/**
		 * 指定された列をキーに行を並べ替えます。
		 * @see [HTML Standard]{@link https://html.spec.whatwg.org/multipage/tables.html#table-sorting-algorithm}
		 * @param {HTMLTableHeaderCellElement} th
		 */
		sort: function (th) {
			var table = th.closest('table');
			var event = new Event('sort', {
				cancelable: true,
			});
			table.dispatchEvent(event);
			if (!event.defaultPrevented) {
				/**
				 * すでに並び替えられている列。
				 * @type {HTMLTableHeaderCellElement}
				 */
				var sortedTH = th.parentElement.querySelector('[sorted]');
				
				/**
				 * 並べ替え後、降順になるなら真。
				 * @type {boolean}
				 */
				var reversed = sortedTH === th && this.getSorting(sortedTH) === 'asc';
				
				// 行リストを配列化
				var tBody = table.tBodies[0];
				var rows = Array.from(tBody.rows);
	
				if (sortedTH === th) {
					// 選択された列が、すでに並べ替えられている列なら
					// sorted属性の設定
					sortedTH.setAttribute('sorted', (reversed ? 'reversed ' : '') + '1');
					// 並び順を反転
					rows.reverse();
				} else {
					// 他の列のsorted属性を削除
					sortedTH.removeAttribute('sorted');
	
					// sorted属性の設定
					th.setAttribute('sorted', '1');
	
					// 昇順に並び替え
					rows.sort((a, b) => this.compareRows(th, a, b));
				}
				
				// 画面に反映
				tBody.removeAttribute('aria-live');
				for (var row of rows) {
					tBody.appendChild(row);
				}
				window.setTimeout(function () {
					tBody.setAttribute('aria-live', 'polite');
				}, 0);
			}
		},
		
		/**
		 * 行の挿入位置を取得します。
		 * @param {HTMLTableElement} table
		 * @param {HTMLTableRowElement} row
		 * @returns {number} 0から始まるインデックス。
		 */
		getInsertPosition: function (table, row) {
			var insertingColumn = table.querySelector('[sorted]');
			var reversed = this.getSorting(insertingColumn) === 'desc';
			var rows = table.tBodies[0].rows;
			var insertPosition = rows.length;
			for (var comparisonRow of Array.from(rows)) {
				var 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) {
			var value;
			var cell = row.cells[th.cellIndex];
			var 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) {
			var 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) {
			var targetIndex = th.cellIndex;
			var refIndex = typeof refTH === 'number' ? refTH : (refTH ? refTH.cellIndex : -1);
			for (var tr of this.getRows(th.closest('table'))) {
				tr.insertBefore(tr.cells[targetIndex], tr.cells[refIndex]);
			}
			
			// 表示する列の設定項目の並び替え
			var 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') {
				var targetIndex = th.cellIndex;
				for (var 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') {
				var targetIndex = th.cellIndex;
				for (var tr of this.getRows(th.closest('table'))) {
					tr.cells[targetIndex].removeAttribute('aria-hidden');
				}
			}
		},
		
		/**
		 * 列の順番を取得します。
		 * @return {string[]}
		 */
		getColumnPositions: function () {
			return Array.from(document.querySelectorAll('#programs th')).map(function (th) {
				return th.id;
			});
		},
	
		/**
		 * 列の移動先を示すクラス名を削除します。
		 */
		removeOldClassName: function () {
			var oldRef = document.querySelector('.inserting-before, .inserting-after');
			if (oldRef) {
				oldRef.classList.remove('inserting-before', 'inserting-after');
			}
		},
		
		/**
		 * 列の順番を反映します。
		 * @param {?string} version
		 * @param {string[]} columnPositions - 指定されなかった列は末尾に並びます。
		 */
		reflectColumnPositions: function (version, columnPositions) {
			if (!version) {
				// 5.0.0 より前のバージョンの設定であれば
				columnPositions.unshift('service');
				GM_setValue('columns-position', JSON.stringify(columnPositions));
			}
			
			var tBody = document.querySelector('#programs tbody');
			tBody.removeAttribute('aria-live');
			var i = 0;
			for (var column of columnPositions) {
				this.moveColumn(document.getElementById(column), i);
				i++;
			}
			window.setTimeout(function () {
				tBody.setAttribute('aria-live', 'polite');
			}, 0);
		},
	}
	
	/**
	 * ユーザー設定値、およびその変更に関するメソッド、プロパティ。
	 */
	window.UserSettings = {
		/**
		 * 省略設定が有効の時に、一項目で表示する最大の符号単位数。
		 * @constant {number}
		 */
		MAX_VISIBLE_CHARACTERS: 60,
		
		/**
		 * 表示を省略した際に、ヒットした文字列より前に表示する最大の符号単位数。
		 * @constant {number}
		 */
		MAX_BEFORE_CHARACTERS: 3,
		
		/**
		 * 表示を省略した際に、ヒットした文字列より前に表示する最大の符号単位数。
		 * @constant {number}
		 */
		MAX_BEFORE_CHARACTERS: 3,
		
		/**
		 * 検索条件。
		 * @type {SearchCriteria[]}
		 */
		words: [],
		
		/**
		 * 検索から除外するユーザー、コミュニティ、チャンネルのURL。exclusionsFromExternalを含む。
		 * @type {string[]}
		 */
		exclusions: [],
		
		/**
		 * ユーザー設定値 NGsURI から取得した検索から除外するユーザー、コミュニティ、チャンネルのURL。
		 * @type {string[]}
		 */
		exclusionsFromExternal: [],
		
		/**
		 * ユーザー設定値をJSONファイルにエクスポートします。
		 */
		export: function () {
			var exportedValues = {};
			for (var key in UserSettings.schema.properties) {
				var property = UserSettings.schema.properties[key];
				var value;
				switch (property.type) {
					case 'string':
					case 'integer':
					case 'boolean':
						value = key === 'audioData' ? UserSettings.getLargeValue(key) : GM_getValue(key);
						break;
					case 'number':
						value = GM_getValue(key);
						if (value !== undefined && value !== null) {
							value = Number.parseFloat(value);
						}
						break;
					case 'array':
					case 'object':
						value = 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>`
			);
			var 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="" />`
			);
			var input = document.body.lastElementChild;
			
			input.addEventListener('change', function parse(event) {
				event.target.removeEventListener(event.type, parse);
				var file = event.target.files[0];
				if (file) {
					var reader = new FileReader();
					reader.addEventListener('load', function (event) {
						var 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 (var key in result) {
									if (result[key] === null) {
										delete result[key];
									}
								}
							}
							var 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) {
									GM_setValue('NGsURI', result.NGsURI);
								} else {
									GM_deleteValue('NGsURI');
								}
								
								// 行のソート
								if (!result.order) {
									result.order = UserSettings.schema.properties.order.default;
								}
								var sortedTH = document.querySelector('#programs [sorted]');
								if (result.order.name !== sortedTH.id) {
									sortedTh = document.getElementById(result.order.name);
									sortedTh.click();
								}
								if (result.order.order !== TableProcessor.getSorting(sortedTH)) {
									sortedTh.click();
								}
								
								// 列の位置
								if (result['columns-position']) {
									GM_setValue('columns-position', JSON.stringify(result['columns-position']));
									TableProcessor.reflectColumnPositions(result.version, result['columns-position']);
								} else {
									GM_deleteValue('columns-position');
									TableProcessor.reflectColumnPositions(GM_info.script.version, UserSettings.schema.properties['columns-position'].default);
								}
								
								// 表示される列
								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 (var key of ['exclusionMemberOnly', 'ellipsisTooLongRSSData', 'languageFilter']) {
									var input = document.getElementsByName(key)[0];
									if (input.checked !== (key in result ? result[key] : UserSettings.schema.properties[key].default)) {
										input.click();
									}
								}
								
								// ミュート
								var alertTone = document.getElementById('alert-tone');
								if (result.audioMuted) {
									alertTone.muted = true;
								}
								
								// 音量
								if ('audioVolume' in result) {
									alertTone.volume = result.audioVolume;
								}
								
								// 音声ファイル
								var deleteSound = document.getElementsByName('delete-sound')[0];
								if (!deleteSound.hidden) {
									deleteSound.click();
								}
								if (result.audioData) {
									var 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。
		 */
		setAudioData: function (audioData) {
			try {
				UserSettings.setLargeValue('audioData', audioData);
				var 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', e));
				} else {
					throw error;
				}
			}
		},
		
		/**
		 * 検索語句文字列を {@link SearchCriteria} に変換します。
		 * @param {string[]} words - 正規化済みの文字列。
		 * @returns {SearchCriteria[]}
		 */
		parseWords: function (words) {
			return words.map(function (word) {
				var searchCriteria = {
					plus: [],
					minus: [],
				};
				for (var value of StringProcessor.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) {
			var url = null;
			if (exclusion.startsWith('http')) {
				url = exclusion;
			} else {
				var result = /(?:co|ch)[1-9][0-9]*/.exec(exclusion);
				if (result) {
					url = 'http://com.nicovideo.jp/community/' + result[0];
				}
			}
			return url;
		},
		
		/**
		 * 検索対象のサービスを取得します。
		 * @return {string[]}
		 */
		getTargetServices: function () {
			return Array.from(document.querySelectorAll('[name="target-services"]:checked')).map(checkbox => checkbox.value);
		},
		
		/**
		 * 表示中の列を取得します。
		 * @return {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) {
			for (var service of document.getElementsByName('target-services')) {
				if ((services.indexOf(service.value) !== -1) !== service.checked) {
					service.click();
				}
			}
		},
		
		/**
		 * 最後に検索結果の取得に成功にした日時を表示します。
		 * @param {Service} service
		 */
		showLatestUpdatedDate: function (service) {
			var date = new Date();
			var 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) {
			var message = error.toString();
			if ('lineNumber' in error) {
				message += ` (${error.lineNumber}:${error.columnNumber})`
			}
			var 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
		 */
		showColumns: function (version, columns) {
			if (!version) {
				// 5.0.0 より前のバージョンの設定であれば
				columns.unshift('service');
				if (columns.indexOf('category') === -1) {
					columns.push('category');
				}
				GM_setValue('visible-columns', JSON.stringify(columns));
			}
			
			for (var 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 {string}
		 * @see [GM_setValue size exception(1 * 1024 * 1024) · Issue #1 · Constellation/ldrfullfeed · GitHub]{@link https://github.com/Constellation/ldrfullfeed/issues/1}
		 */
		setLargeValue: function (name, value) {
			var error;
			GM_setValue(name, value);
			if (GM_getValue(name) !== value) {
				// 値が正しく設定されていなければ
				var item = this.getValuesFromLocalStorage();
				item[name] = value;
				window.localStorage.setItem(Alert.ID, JSON.stringify(item));
				GM_deleteValue(name);
			}
			return value;
		},
		
		/**
		 * {@link UserSettings.setLargeValue} で保存したデータを取得します。
		 * @param {type} name
		 * @param {*} defaultValue
		 * @returns {*}
		 */
		getLargeValue: function (name, defaultValue) {
			var value = GM_getValue(name);
			if (value === undefined || value === null) {
				var item = this.getValuesFromLocalStorage();
				value = item[name];
			}
			return value === undefined ? defaultValue : value;
		},
		
		/**
		 * {@link UserSettings.setLargeValue} で保存したデータを削除します。
		 * @param {string} name
		 */
		deleteLargeValue: function (name) {
			GM_deleteValue(name);
			var 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 () {
			var item = window.localStorage.getItem(Alert.ID);
			if (item) {
				try {
					item = JSON.parse(item);
				} catch (e) {
					item = {};
				}
			} else {
				item = {};
			}
			return item;
		}
	};
	
	/**
	 * 文字列の処理を行うユーティリティークラス。
	 */
	window.StringProcessor = {
		/**
		 * ひらがなをカタカナに変換するときの加数。
		 * @constant {number}
		 */
		ADDEND_HIRAGANA_TO_KATAKANA: 'ァ'.charCodeAt() - 'ぁ'.charCodeAt(),
		
		/**
		 * ひらがなをカタカナに変換します。
		 * @param {string} str
		 * @returns {string}
		 */
		convertToKatakana: function (str) {
			return str.replace(
				/[ぁ-ゖ]/g,
				match => String.fromCharCode(match.charCodeAt() + this.ADDEND_HIRAGANA_TO_KATAKANA)
			);
		},
		
		/**
		 * 正規化し、連続する空白文字を半角スペースに。
		 * @param {string} str
		 * @returns {string}
		 */
		normalize: function (str) {
			return str.normalize('NFKC').replace(/\s+/g, ' ');
		},
		
		/**
		 * 文字種を統一します。
		 * @param {string} str - 正規化済みの文字列。
		 * @returns {string}
		 */
		unifyCases: function (normalizedStr) {
			return this.convertToKatakana(normalizedStr).toLocaleLowerCase();
		},
	};
	
	var WordProcessor = {
		/**
		 * OR検索を行います。
		 * @param {string} str
		 * @param {SearchCriteria[]} words
		 * @return {?SearchCriteria}
		 */
		orSearch: function (str, words) {
			var searchCriteria = null;
			for (var word of words) {
				if (this.andSearch(str, word)) {
					searchCriteria = word;
					break;
				}
			}
			return searchCriteria;
		},
		
		/**
		 * AND検索を行います。
		 * @param {string} str
		 * @param {SearchCriteria} word
		 * @return {boolean}
		 */
		andSearch: function (str, word) {
			return this.minusSearch(str, word.minus) && word.plus.every(plus => p(str).includes(plus));
		},
		
		/**
		 * マイナス検索を行います。
		 * @param {string} str
		 * @param {string[]} minusWord
		 * @return {boolean} str に minusWord のどの文字列も含まれていなければ真。
		 */
		minusSearch: function (str, minusWord) {
			return !minusWord.some(minus => p(str).includes(minus));
		},
		
		/**
		 * 検索語句が含まれる位置を取得します。
		 * @param {string} data
		 * @param {(SearchCriteria|SearchCriteria[])} searchCriteria
		 * @returns {number[][]} [先頭のオフセット, 末尾のオフセット] の配列。重なる部分は一つにまとめます。
		 */
		getMatches: function (data, searchCriteria) {
			var words = Array.isArray(searchCriteria)
				? searchCriteria.reduce((words, searchCriteria) => words.concat(searchCriteria.plus), [])
				: searchCriteria.plus;
			
			var unified = StringProcessor.unifyCases(data);
			var offsets = [];
			for (var word of words) {
				var index = unified.indexOf(word);
				if (index !== -1) {
					offsets.push([index, index + word.length]);
				}
			}
	
			// ソート
			offsets.sort((a, b) => a[0] - b[0]);
	
			// 位置が重なっていたら一つにまとめる
			for (var 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)} 
		 */
		convertOffsetsToRanges: function (text, offsets) {
			return offsets.map(function (offset) {
				var range = new Range();
				range.setStart(text, offset[0]);
				range.setEnd(text, offset[1]);
				return range;
			});
		},
		
		/**
		 * 指定された部分を括弧で囲みます。
		 * @param {string} data
		 * @param {number[][]} offsets - 昇順に並んでいる、重なる部分が無い [先頭のオフセット, 末尾のオフセット] の配列。
		 * @returns {string}
		 */
		markMatchesAsPlainText: function (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」を含む配列。
		 */
		extractTextNode: function (text, offsets, maxLength, beforeLength) {
			/**
			 * 切り取り範囲。
			 * @type {Range[]}
			 */
			var trimRanges = [];
			
			/**
			 * 切り取る前の文字列。
			 * @type {number}
			 */
			var data = text.data;
			
			/**
			 * 切り取る前の文字列の符号単位数。
			 * @type {number}
			 */
			var dataLength = text.length;
			
			/**
			 * 戻り値。
			 * @type {string[]}
			 */
			var classes = [];
				
			/**
			 * 表示する部分の終了位置。
			 * @type {number}
			 */
			var viewEndOffset;
			
			if (offsets.length > 0) {
				// 検索語句が一致する箇所があれば
				/**
				 * 表示する部分の開始位置。
				 * @type {number}
				 */
				var 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) {
					// 表示部分の開始位置が先頭より後ろなら
					var charCode = data.charCodeAt(viewStartOffset);
					if (0xDC00 <= charCode && charCode <= 0xDFFF) {
						// 表示部分の先頭文字が下位サロゲートであれば
						viewStartOffset--;
					}
					if (viewStartOffset > 0) {
						var range = new Range();
						range.setStart(text, 0);
						range.setEnd(text, viewStartOffset);
						trimRanges.push(range);
						classes.push('ellipsis-left');
					}
				}
			} else {
				viewEndOffset = maxLength;
			}
			
			if (viewEndOffset < dataLength) {
				// 表示部分の終了位置が末尾より前なら
				var charCode = data.charCodeAt(viewEndOffset - 1);
				if (0xD800 <= charCode && charCode <= 0xDBFF) {
					// 表示部分の末尾文字が上位サロゲートであれば
					viewEndOffset++;
				}
				if (viewEndOffset < dataLength) {
					var range = new Range();
					range.setStart(text, viewEndOffset);
					range.setEnd(text, dataLength);
					trimRanges.push(range);
					classes.push('ellipsis-right');
				}
			}
			
			// 切り取る
			for (var range of trimRanges) {
				range.deleteContents();
			};
			
			return classes;
		},
	};
	
	/**
	 * 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}
	  */
	
	/**
	 * ライブ配信サービス。
	 * @class
	 * @augments EventTarget
	 * @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秒。
	 * @global
	 */
	function Service (details) {
		/**
		 * 情報を取得する間隔の既定値。ミリ秒。
		 * @constant {number}
		 * @access private
		 */
		this.DEFAULT_DELAY = 6 * DateUtils.MINUTES_TO_MILISECONDS;
		
		/**
		 * 結果が複数ページにわたる場合に、次のページを取得するまでの間隔の既定値。ミリ秒。
		 * @constant {number}
		 * @access private
		 */
		this.DEFAULT_PAGING_DELAY = 1 * DateUtils.SECONDS_TO_MILISECONDS;
		
		/**
		 * サービスを識別する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;
	
		/**
		 * 検索ワードにヒットした番組を繰り返し取得します。
		 * @param {HTTPRequestInit} [nextSearchInit] - 次のページの取得に必要な情報。
		 * @returns {Promise}
		 * @fires Service#ProgramEvent:progress
		 * @fires Service#LoadedEvent:load
		 * @access private
		 */
		this.getHitPrograms = function (nextSearchInit) {
			if (this.enabled) {
				/**
				 * 現在の検索ワード。接続前と取得完了時の検索ワードの変化を防ぎます。
				 * @type {SearchCriteria[]}
				 */
				var words = UserSettings.words;
				
				var searchInit = nextSearchInit
					|| details.getDetails(details.disabledOr ? words[this.wordIndex] : words);
				searchInit.timeout = Service.TIMEOUT;
				this.request = new HTTPRequest(searchInit);
				
				this.request.send().then(response => {
					var programs = details.parseResponse(response, searchInit, words);
					if (programs instanceof Error) {
						return Promise.reject(error);
					}
					
					// 番組情報の取得に成功していれば
					for (var program of Array.from(programs.programs)) {
						var entry = details.convertIntoEntry(program);
						entry.service = this;
						var hit = false;
						if (details.disabledOr) {
							// OR検索ができないサービスなら
							entry.searchCriteria = words[this.wordIndex];
						}
						if (details.disabledSearch) {
							// 全件取得が可能なサービスなら
							hit = WordProcessor.orSearch(entry.getSearchTarget(), words);
							if (hit) {
								entry.searchCriteria = hit;
							}
						} else if (details.disabledMinus) {
							// マイナス検索ができないサービスなら
							hit = WordProcessor.minusSearch(entry.getSearchTarget(), entry.searchCriteria.minus);
						} else {
							hit = true;
						}
						// 言語の絞り込み
						if (hit && document.getElementsByName('languageFilter')[0].checked && details.disabledLanguageFilter
							&& window.navigator.language.split('-')[0] !== entry.language.split('-')[0]) {
							hit = false;
						}
						if (hit) {
							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), details.pagingDelay || this.DEFAULT_PAGING_DELAY, programs.next);
					} else {
						// 取得完了
						this.dispatchEvent(new CustomEvent('load', { detail: {
							service: this,
							programs: this.currentPrograms,
							searchCriteria: details.disabledOr ? words[this.wordIndex] : null,
						}}));
						this.currentPrograms = [];
						
						if (details.disabledOr) {
							// OR検索ができないサービスなら
							this.wordIndex++;
							if (!UserSettings.words[this.wordIndex]) {
								// 次の検索語句が存在しなければ
								this.wordIndex = 0;
							}
						}
						
						this.timer = window.setTimeout(this.getHitPrograms.bind(this), details.delay || this.DEFAULT_DELAY);
					}
				}).catch(error => {
					// 番組情報の取得に失敗していれば
					if (!(error instanceof AbortException)) {
						// 意図的な停止による例外でなければ
						console.log(error);
						UserSettings.showLatestError(this, error);
						// 一定時間後に最初から検索をやり直す
						this.timer = window.setTimeout(this.getHitPrograms.bind(this), Service.RETRY_DELAY);
					}
				});
			}
		};
		
		// EventTargetの疑似継承
		// <http://stackoverflow.com/questions/22186467/how-to-use-javascript-eventtarget#answer-24216547>
		var eventTarget = new DocumentFragment();
		for (var key in this) {
			eventTarget[key] = typeof this[key] === 'function' ? this[key].bind(this) : this[key];
		}
		for (var methodName of ['addEventListener', 'removeEventListener', 'dispatchEvent']) {
			this[methodName] = eventTarget[methodName].bind(eventTarget);
		}
	};
	
	/**
	 * タイムアウトミリ秒数。
	 * @constant {number}
	 */
	Service.TIMEOUT = 10 * DateUtils.SECONDS_TO_MILISECONDS;
	
	/**
	 * メンテナンス中など、サーバー側のエラーが発生した場合に再度取得する間隔。ミリ秒。
	 * @constant {number}
	 */
	Service.RETRY_DELAY = 15 * DateUtils.MINUTES_TO_MILISECONDS;
	
	/**
	 * 検索を開始します。
	 */
	Service.prototype.start = function () {
		if (!this.enabled) {
			// 検索が無効であれば
			this.enabled = true;
			this.getHitPrograms();
		}
	};
	
	/**
	 * 検索を停止します。
	 */
	Service.prototype.stop = function () {
		this.enabled = false;
		if (this.request) {
			this.request.abort();
		}
		window.clearTimeout(this.timer);
	};
	
	/**
	 * 検索語句を読み込み直して最初から検索しなおします。
	 */
	Service.prototype.reset = function () {
		if (this.disabledOr) {
			this.wordIndex = 0;
			TableProcessor.removeProgramsWithService(this);
		}
	};
	
	/**
	 * 1つのライブ配信番組。
	 * @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} [details.searchCriteria] - 検索条件。
	 */
	window.Program = function (link, title, otherDetails, searchCriteria) {
		this.link = link;
		this.title = StringProcessor.normalize(title);
		for (var key in otherDetails) {
			var value = otherDetails[key];
			if (value !== undefined && value !== null) {
				switch (key) {
					case 'summary':
						value = StringProcessor.normalize(value);
						break;
					case 'categories':
						value = value.map(StringProcessor.normalize);
						break;
					case 'community':
						value.name = StringProcessor.normalize(value.name);
						break;
				}
				this[key] = value;
			}
		}
		this.searchCriteria = searchCriteria;
		
		/**
		 * @type {Service}
		 */
		this.service;
	};
	
	/**
	 * 配信者、またはコミュニティ。
	 * @typedef {Object} Person
	 * @property {string} name - 名前。
	 * @property {string} [url] - URL。
	 */
	
	/**
	 * 検索対象を取得します。
	 * @return {string} 文字種を統一した文字列。
	 */
	Program.prototype.getSearchTarget = function () {
		var target = [this.title];
		if (this.categories) {
			target.concat(this.categories);
		}
		if (this.summary) {
			target.push(this.summary);
		}
		if (this.community) {
			target.push(this.community.name);
		}
		return StringProcessor.unifyCases(target.join(' '));
	};
}

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

/**
 * ライブラリの定義、ECMAScriptとWHATWG仕様のPolyfill、prototype汚染回避など。
 */
function polyfill() {
	/**
	 * DOM関連のメソッド等。
	 * @version 1.0.0
	 */
	window.DOMUtils = {
		/**
		 * Atom名前空間。
		 * @constant {string}
		 */
		ATOM_NAMESPACE: 'http://www.w3.org/2005/Atom',
		
		/**
		 * XMLの特殊文字を文字参照に置換します。
		 * @param {string} str - プレーンな文字列。
		 * @returns {string} HTMLとして扱われる文字列。
		 */
		convertSpecialCharactersToCharacterReferences: function (str) {
			return String(str).replace(/[&<>"']/g, function (specialCharcter) {
				return '&#x' + specialCharcter.charCodeAt(0).toString(16) + ';'
			});
		},
		
		/**
		 * テンプレート文字列のタグとして用いることで、式内にあるXMLの特殊文字を文字参照に置換します。
		 * @param {string[]} htmlTexts
		 * @param {...string} plainText
		 * @returns {string} HTMLとして扱われる文字列。
		 */
		escapeTemplateStrings() {
			return String.raw.apply(null, Array.prototype.map.call(arguments, function (plainText, i) {
				return i === 0 ? plainText : this.convertSpecialCharactersToCharacterReferences(plainText);
			}, this));
		},
	};
	
	/**
	 * {@link DOMUtils.escapeTemplateStrings}、または {@link DOMUtils.convertSpecialCharactersToCharacterReferences} の短縮表記。
	 */
	window.h = function () {
		return Array.isArray(arguments[0])
			? DOMUtils.escapeTemplateStrings.apply(DOMUtils, arguments)
			: DOMUtils.convertSpecialCharactersToCharacterReferences(arguments[0]);
	};
	
	/**
	 * HTTPリクエスト。
	 * @param {HTTPRequestInit} init
	 * @constructor
	 * @version 1.0.0
	 */
	window.HTTPRequest = function (init) {
		/** @access private */
		this.details = init;
		
		/**
		 * @param {Object} client
		 * @param {Function} resolve
		 * @param {Function} reject
		 * @param {string} [responseType]
		 * @access private
		 */
		this.onload = function (client, resolve, reject, responseType) {
			if (client.status === 200) {
				var response;
				var errorMessage;
				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':
							var 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'
								);
								
								var 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 {
				reject(new ErrorStatusException(client.status));
			}
		};
		
		/**
		 * @param {Function} reject
		 * @access private
		 */
		this.ontimeout = function (reject) {
			reject(new TimeoutException());
		};
		
		/**
		 * @param {Function} reject
		 * @access private
		 */
		this.onabort = function (reject) {
			reject(new AbortException());
		};
	};
	
	/**
	 * HTTPリクエストに必要な情報。
	 * @typedef {Object} HTTPRequestInit
	 * @see [GM_xmlhttpRequest - GreaseSpot Wiki]{@link http://wiki.greasespot.net/GM_xmlhttpRequest#Arguments}
	 * @see [Fetch Standard (日本語訳)]{@link http://www.hcn.zaq.ne.jp/___/WEB/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」を指定する。
	 */
	
	/**
	 * リクエストを送信します。
	 * @returns {Promise.<string|Object>}
	 */
	HTTPRequest.prototype.send = function () {
		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);
				});
				if (this.details.headers) {
					for (var 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 {
				var details = {};
				for (var 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);
				};
				this.client = GM_xmlhttpRequest(details);
			}
		});
	};
	
	/**
	 * リクエストを取り消します。
	 */
	HTTPRequest.prototype.abort = function () {
		if (this.client) {
			this.client.abort();
		}
	};
	
	/**
	 * @param {string} message
	 * @constructor
	 * @see [Custom Error Types | Error - JavaScript | MDN]{@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Error#Custom_Error_Types}
	 */
	window.ConnectionException = function (message) {
		this.name = 'ConnectionException';
		this.message = message || 'Connection exception occured.';
		this.stack = (new Error()).stack;
	}
	ConnectionException.prototype = Object.create(Error.prototype);
	ConnectionException.prototype.constructor = ConnectionException;
	
	/**
	 * @param {string} message
	 * @constructor
	 */
	window.TimeoutException = function (message) {
		this.name = 'TimeoutException';
		this.message = message || 'Connection timed out.';
		this.stack = (new ConnectionException()).stack;
	}
	TimeoutException.prototype = Object.create(ConnectionException.prototype);
	TimeoutException.prototype.constructor = TimeoutException;
	
	/**
	 * @param {number} code - HTTPステータスコード。
	 * @param {string} [message]
	 * @constructor
	 */
	window.ErrorStatusException = function (code, message) {
		this.name = 'ErrorStatusException';
		this.message = message || ('HTTP status-code was %s.').replace('%s', code);
		this.stack = (new ConnectionException()).stack;
		/**
		 * HTTPステータスコード。
		 * @type {number}
		 */
		this.code = code;
	}
	ErrorStatusException.prototype = Object.create(ConnectionException.prototype);
	ErrorStatusException.prototype.constructor = ErrorStatusException;
	
	/**
	 * @param {string} message
	 * @constructor
	 */
	window.AbortException = function (message) {
		this.name = 'AbortException';
		this.message = message || 'Request was aborted.';
		this.stack = (new ConnectionException()).stack;
	}
	AbortException.prototype = Object.create(ConnectionException.prototype);
	AbortException.prototype.constructor = AbortException;

	/**
	 * 時間に関するユーティリティクラス。
	 * @version 1.0.0
	 */
	window.DateUtils = {
		/**
		 * 日をミリ秒に変換するときの乗数。
		 * @constant {number}
		 */
		DAYS_TO_MILISECONDS: 24 * 60 * 60 * 1000,
		
		/**
		 * 時間をミリ秒に変換するときの乗数。
		 * @constant {number}
		 */
		HOURS_TO_MILISECONDS: 60 * 60 * 1000,
		
		/**
		 * 分をミリ秒に変換するときの乗数。
		 * @constant {number}
		 */
		MINUTES_TO_MILISECONDS: 60 * 1000,
		
		/**
		 * 秒をミリ秒に変換するときの乗数。
		 * @constant {number}
		 */
		SECONDS_TO_MILISECONDS: 1000,
		
		/**
		 * 現在時刻から指定した時刻を引いた差を返します。
		 * @param {Date} value
		 * @returns {Object.<string>} dateTimeプロパティに ISO 8601 形式の文字列 (負になる場合は「PT0S」)、textプロパティに「○時間○分」のような形式の文字列。
		 */
		getDuration: function (value) {
			var milliseconds = Date.now() - value.getTime();
			var minutes = Math.round(milliseconds / this.MINUTES_TO_MILISECONDS);
			var sign = minutes >= 0 ? 1 : -1;
			minutes = Math.abs(minutes);
			var hours = Math.floor(minutes / 60);
			minutes = minutes % 60;
			return {
				dateTime: `PT${sign === -1 ? 0 : milliseconds / this.SECONDS_TO_MILISECONDS}S`,
				text: hours ? _('%d 時間 %u 分').replace('%d', sign * hours).replace('%u', minutes) : _('%d 分').replace('%d', sign * minutes),
			};
		},
		
		/**
		 * ISO 8601 形式の文字列からミリ秒数を取得します。
		 * @param {string} dateTime
		 * @returns {?number}
		 */
		parseDurationString: function (dateTime) {
			var duration = null;
			var result = /^P([0-9]+D)?(?:T([0-9]+H)?([0-9]+M)?([0-9]+(?:\.[0-9]{0,3})?S)?)?$/.exec(dateTime);
			if (result) {
				duration = 0;
				if (result[1]) {
					duration += Number.parseInt(result[1]) * this.DAYS_TO_MILISECONDS;
				}
				if (result[2]) {
					duration += Number.parseInt(result[2]) * this.HOURS_TO_MILISECONDS;
				}
				if (result[3]) {
					duration += Number.parseInt(result[3]) * this.MINUTES_TO_MILISECONDS;
				}
				if (result[4]) {
					duration += Number.parseFloat(result[4]) * this.SECONDS_TO_MILISECONDS;
				}
			}
			return duration;
		},
		
		/**
		 * 日本標準時の時刻文字列 (hh:mm) をDateインスタンスに変換します。
		 * @param {string} time - 過去を表す時刻。
		 * @returns {Date}
		 */
		parseJSTString: function (time) {
			/**
			 * 日本標準時 (+09:00) の時間帯 (UTCとの差)。ミリ秒。
			 * @constant {number}
			 */
			var TIMEZONE_JST = 9 * this.HOURS_TO_MILISECONDS;
			
			var hoursAndMinutes = time.split(':');
			var date = new Date();
			date.setUTCHours(Number.parseInt(hoursAndMinutes[0]), Number.parseInt(hoursAndMinutes[1]), 0, 0);
			date.setTime(date.getTime() - TIMEZONE_JST);
			if (date.getTime() > Date.now()) {
				date.setTime(date.getTime() - this.DAYS_TO_MILISECONDS);
			}
			return date;
		},
	};
	
	// For Firefox
	if (typeof cloneInto !== 'undefined' && typeof Proxy !== 'undefined' /* exclude Tampermonkey*/) {
		CustomEvent = new Proxy(CustomEvent, { construct: function (target, args) {
			if (args.length >= 2
				&& typeof args[1] === 'object' &&  typeof args[1].detail === 'object' && args[1].detail !== null) {
				args[1].detail = cloneInto(args[1].detail, window, {
					cloneFunctions: true,
					wrapReflectors: true,
				});
			}
			return eval('new target(...args)');
		} });
	}
	
	/**
	 * Webページ側のコンテキストでなければprototype拡張ができない問題に対処します。
	 * @param {*} value - prototype拡張を行ったインターフェースのインスタンス。
	 * @returns {*} FirefoxであればProxyインスタンス。
	 * @version 1.0.0
	 */
	window.p = function (value) {
		var element;
		if (typeof MozSettingsEvent !== 'undefined') {
			switch (typeof value) {
				case 'string':
					value = new String(value);
					break;
				case 'number':
					value = new Number(value);
					break;
				case 'boolean':
					value = new Boolean(value);
			}
			value = new Proxy(value, {
				get: function (target, name) {
					var value;
					if (name === 'dataset' && target instanceof HTMLElement) {
						// DOMStringMap deleter の実装
						element = target;
						value = new Proxy(target.dataset, {
							deleteProperty: function (dataset, name) {
								element.removeAttribute(
									'data-' + name.replace(/[A-Z]/g, upper => '-' + upper.toLowerCase())
								);
								return true;
							},
						});
					} else if (name in target) {
						value = target[name];
					} else {
						for (var proto = target.__proto__; proto; proto = proto.__proto__) {
							if (name in proto) {
								var descriptor = Object.getOwnPropertyDescriptor(proto, name);
								value = descriptor.value || descriptor.get.call(target);
								break;
							}
						}
					}
					return typeof value === 'function' ? value.bind(target) : value;
				},
				set: function (target, name, value) {
					if (name in target) {
						target[name] = value;
					} else {
						var set = false;
						for (var proto = target.__proto__; proto; proto = proto.__proto__) {
							if (name in proto) {
								Object.getOwnPropertyDescriptor(proto, name).set.call(target);
								set = true;
								break;
							}
						}
						if (!set) {
							target[name] = value;
						}
					}
					return true;
				},
			});
		}
		return value;
	};
	
	// Polyfill for Firefox, Opera, and Google Chrome
	if (!('createFor' in URL)) {
		/** @see [Bug 1062917 - Implement URL.createFor]{@link https://bugzilla.mozilla.org/show_bug.cgi?id=1062917} */
		Object.defineProperty(URL, 'createFor', {
			writable: true,
			enumerable: false,
			configurable: true,
			value: function (blob) {
				/**
				 * 分をミリ秒に変換するときの乗数。
				 * @constant {number}
				 */
				var MINUTES_TO_MILISECONDS = 60 * 1000;
				
				/**
				 * Blob URL を自動破棄するまでのミリ秒数。
				 * @constant {number}
				 */
				var MAX_LIFETIME = 10 * MINUTES_TO_MILISECONDS;
				
				var url = this.createObjectURL(blob);
				window.setTimeout(this.revokeObjectURL.bind(this), MAX_LIFETIME, url);
				return url;
			},
		});
	}
	
	// Polyfill for Firefox 38 ESR, Opera, and Google Chrome
	try {
		new DragEvent('drag');
	} catch (e) {
		/**
		 * @constructor
		 * @param {string} type
		 * @param {DragEventInit} [eventInitDict]
		 * @see [Bug 1135627 - DragEvent constructor throws "TypeError: Illegal constructor."]{@link https://bugzilla.mozilla.org/show_bug.cgi?id=1135627}
		 * @see [Issue 498504 - chromium - Implement DragEvent and move MouseEvent.dataTransfer to DragEvent]{@link https://code.google.com/p/chromium/issues/detail?id=498504}
		 * @see [The DragEvent interface]{@link https://html.spec.whatwg.org/multipage/interaction.html#the-dragevent-interface}
		 * @name DragEvent
		 */
		Object.defineProperty(window, 'DragEvent', {
			writable: true,
			enumerable: false,
			configurable: true,
			value: MouseEvent,
		});
	}
	
	// Polyfill for Firefox 38 ESR
	if (!''.includes) {
		/** @see [Bug 1102219 – Rename String.prototype.contains to String.prototype.includes]{https://bugzilla.mozilla.org/show_bug.cgi?id=1102219} */
		Object.defineProperty(String.prototype, 'includes', {
			writable: true,
			enumerable: false,
			configurable: true,
			value: String.prototype.contains,
		});
	}

	// Polyfill for Opera and Google Chrome
	// prototype汚染が行われる Prototype JavaScript Framework (prototype.js) 1.6.0.3 のバグを修正
	if (window.chrome) {
		Object.defineProperty(Array, 'from', { writable: false });
		Object.defineProperty(Object, 'extend', {
			writable: false,
			value: function (destination, source) {
				for (var property in source) {
					if (property in destination || property === 'toJSON') {
						continue;
					}
					destination[property] = source[property];
				}
				return destination;
			},
		});
	}
	
	if (!(Symbol.iterator in NodeList.prototype)) {
		Object.defineProperties(NodeList.prototype, /** @lends NodeList# */ {
			/**
			 * @see [Issue 401699 - chromium - Add iterator support to NodeList and friends]{@link https://code.google.com/p/chromium/issues/detail?id=401699}
			 * @version 1.1.0
			 * @returns {Iterator.<Array.<number, Node>>}
			 * @name NodeList#@@iterator
			 */
			[Symbol.iterator]: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function* () {
					for (var i = 0, l = this.length; i < l; i++) {
						yield this[i];
					}
				}
			},
			/**
			 * @returns {Iterator.<Array.<number, Node>>}
			 * @function
			 */
			entries: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function* () {
					for (var i = 0, l = this.length; i < l; i++) {
						yield [i, this[i]];
					}
				}
			},
			/**
			 * @returns {Iterator.<number>}
			 * @function
			 */
			keys: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function* () {
					for (var i = 0, l = this.length; i < l; i++) {
						yield i;
					}
				}
			},
			/**
			 * @returns {Iterator.<Node>}
			 * @function
			 */
			values: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function* () {
					for (var i = 0, l = this.length; i < l; i++) {
						yield this[i];
					}
				}
			},
		});
	}
	
	if (!CSS.escape) {
		var global = window;
		/*! https://mths.be/cssescape v1.1.0 by @mathias | MIT license */
		;(function(root) {
		
			if (!root.CSS) {
				root.CSS = {};
			}
		
			var CSS = root.CSS;
		
			var InvalidCharacterError = function(message) {
				this.message = message;
			};
			InvalidCharacterError.prototype = new Error;
			InvalidCharacterError.prototype.name = 'InvalidCharacterError';
		
			if (!CSS.escape) {
				// https://drafts.csswg.org/cssom/#serialize-an-identifier
				CSS.escape = function(value) {
					var string = String(value);
					var length = string.length;
					var index = -1;
					var codeUnit;
					var result = '';
					var firstCodeUnit = string.charCodeAt(0);
					while (++index < length) {
						codeUnit = string.charCodeAt(index);
						// Note: there’s no need to special-case astral symbols, surrogate
						// pairs, or lone surrogates.
		
						// If the character is NULL (U+0000), then throw an
						// `InvalidCharacterError` exception and terminate these steps.
						if (codeUnit == 0x0000) {
							throw new InvalidCharacterError(
								'Invalid character: the input contains U+0000.'
							);
						}
		
						if (
							// If the character is in the range [\1-\1F] (U+0001 to U+001F) or is
							// U+007F, […]
							(codeUnit >= 0x0001 && codeUnit <= 0x001F) || codeUnit == 0x007F ||
							// If the character is the first character and is in the range [0-9]
							// (U+0030 to U+0039), […]
							(index == 0 && codeUnit >= 0x0030 && codeUnit <= 0x0039) ||
							// If the character is the second character and is in the range [0-9]
							// (U+0030 to U+0039) and the first character is a `-` (U+002D), […]
							(
								index == 1 &&
								codeUnit >= 0x0030 && codeUnit <= 0x0039 &&
								firstCodeUnit == 0x002D
							)
						) {
							// https://drafts.csswg.org/cssom/#escape-a-character-as-code-point
							result += '\\' + codeUnit.toString(16) + ' ';
							continue;
						}
		
						if (
							// If the character is the first character and is a `-` (U+002D), and
							// there is no second character, […]
							index == 0 &&
							length == 1 &&
							codeUnit == 0x002D
						) {
							result += '\\' + string.charAt(index);
							continue;
						}
		
						// If the character is not handled by one of the above rules and is
						// greater than or equal to U+0080, is `-` (U+002D) or `_` (U+005F), or
						// is in one of the ranges [0-9] (U+0030 to U+0039), [A-Z] (U+0041 to
						// U+005A), or [a-z] (U+0061 to U+007A), […]
						if (
							codeUnit >= 0x0080 ||
							codeUnit == 0x002D ||
							codeUnit == 0x005F ||
							codeUnit >= 0x0030 && codeUnit <= 0x0039 ||
							codeUnit >= 0x0041 && codeUnit <= 0x005A ||
							codeUnit >= 0x0061 && codeUnit <= 0x007A
						) {
							// the character itself
							result += string.charAt(index);
							continue;
						}
		
						// Otherwise, the escaped character.
						// https://drafts.csswg.org/cssom/#escape-a-character
						result += '\\' + string.charAt(index);
		
					}
					return result;
				};
			}
		
		}(typeof global != 'undefined' ? global : this));
	}


/*
 * jsen
 * https://github.com/bugventure/jsen
 *
 * Copyright (c) 2015 Veli Pehlivanov <bugventure@gmail.com>
 * Licensed under the MIT license <http://opensource.org/licenses/mit>
 */

(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.jsen = f()}})(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
module.exports = require('./lib/jsen.js');
},{"./lib/jsen.js":5}],2:[function(require,module,exports){
'use strict';

function type(obj) {
    var str = Object.prototype.toString.call(obj);
    return str.substr(8, str.length - 9).toLowerCase();
}

function deepEqual(a, b) {
    var keysA = Object.keys(a).sort(),
        keysB = Object.keys(b).sort(),
        i, key;

    if (!equal(keysA, keysB)) {
        return false;
    }

    for (i = 0; i < keysA.length; i++) {
        key = keysA[i];

        if (!equal(a[key], b[key])) {
            return false;
        }
    }

    return true;
}

function equal(a, b) {  // jshint ignore: line
    var typeA = typeof a,
        typeB = typeof b,
        i;

    // get detailed object type
    if (typeA === 'object') {
        typeA = type(a);
    }

    // get detailed object type
    if (typeB === 'object') {
        typeB = type(b);
    }

    if (typeA !== typeB) {
        return false;
    }

    if (typeA === 'object') {
        return deepEqual(a, b);
    }

    if (typeA === 'regexp') {
        return a.toString() === b.toString();
    }

    if (typeA === 'array') {
        if (a.length !== b.length) {
            return false;
        }

        for (i = 0; i < a.length; i++) {
            if (!equal(a[i], b[i])) {
                return false;
            }
        }

        return true;
    }

    return a === b;
}

module.exports = equal;
},{}],3:[function(require,module,exports){
'use strict';

var formats = {};

// reference: http://dansnetwork.com/javascript-iso8601rfc3339-date-parser/
formats['date-time'] = /(\d\d\d\d)(-)?(\d\d)(-)?(\d\d)(T)?(\d\d)(:)?(\d\d)(:)?(\d\d)(\.\d+)?(Z|([+-])(\d\d)(:)?(\d\d))/;
// reference: https://github.com/mafintosh/is-my-json-valid/blob/master/formats.js#L7
formats.uri = /^([a-zA-Z][a-zA-Z0-9+-.]*:){0,1}\/\/[^\s]*$/;
// reference: http://stackoverflow.com/questions/201323/using-a-regular-expression-to-validate-an-email-address#answer-8829363
//            http://www.w3.org/TR/html5/forms.html#valid-e-mail-address (search for 'willful violation')
formats.email = /^[a-zA-Z0-9.!#$%&'*+\/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*$/;
// reference: https://www.safaribooksonline.com/library/view/regular-expressions-cookbook/9780596802837/ch07s16.html
formats.ipv4 = /^(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$/;
// reference: http://stackoverflow.com/questions/53497/regular-expression-that-matches-valid-ipv6-addresses
formats.ipv6 = /^(([0-9a-fA-F]{1,4}:){7,7}[0-9a-fA-F]{1,4}|([0-9a-fA-F]{1,4}:){1,7}:|([0-9a-fA-F]{1,4}:){1,6}:[0-9a-fA-F]{1,4}|([0-9a-fA-F]{1,4}:){1,5}(:[0-9a-fA-F]{1,4}){1,2}|([0-9a-fA-F]{1,4}:){1,4}(:[0-9a-fA-F]{1,4}){1,3}|([0-9a-fA-F]{1,4}:){1,3}(:[0-9a-fA-F]{1,4}){1,4}|([0-9a-fA-F]{1,4}:){1,2}(:[0-9a-fA-F]{1,4}){1,5}|[0-9a-fA-F]{1,4}:((:[0-9a-fA-F]{1,4}){1,6})|:((:[0-9a-fA-F]{1,4}){1,7}|:)|[fF][eE]80:(:[0-9a-fA-F]{0,4}){0,4}%[0-9a-zA-Z]{1,}|::([fF]{4}(:0{1,4}){0,1}:){0,1}((25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])\.){3,3}(25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])|([0-9a-fA-F]{1,4}:){1,4}:((25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])\.){3,3}(25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9]))$/;
// reference: http://stackoverflow.com/questions/106179/regular-expression-to-match-dns-hostname-or-ip-address#answer-3824105
formats.hostname = /^([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9-]{0,61}[a-zA-Z0-9])(\.([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9-]{0,61}[a-zA-Z0-9]))*$/;

module.exports = formats;
},{}],4:[function(require,module,exports){
'use strict';

module.exports = function func() {
    var name = arguments[0] || '',
        args = [].join.call([].slice.call(arguments, 1), ', '),
        lines = '',
        vars = '',
        ind = 1,
        tab = '  ',
        bs = '{[',  // block start
        be = '}]',  // block end
        space = function () {
            return new Array(ind + 1).join(tab);
        },
        push = function (line) {
            lines += space() + line + '\n';
        },
        builder = function (line) {
            var first = line[0],
                last = line[line.length - 1];

            if (be.indexOf(first) > -1 && bs.indexOf(last) > -1) {
                ind--;
                push(line);
                ind++;
            }
            else if (bs.indexOf(last) > -1) {
                push(line);
                ind++;
            }
            else if (be.indexOf(first) > -1) {
                ind--;
                push(line);
            }
            else {
                push(line);
            }

            return builder;
        };

    builder.def = function (id, def) {
        vars += space() + 'var ' + id + (def !== undefined ? ' = ' + def : '') + '\n';

        return builder;
    };

    builder.toSource = function () {
        return 'function ' + name + '(' + args + ') {\n' + vars + '\n' + lines + '\n}';
    };

    builder.compile = function (scope) {
        var src = 'return (' + builder.toSource() + ')',
            scp = scope || {},
            keys = Object.keys(scp),
            vals = keys.map(function (key) { return scp[key]; });

        return Function.apply(null, keys.concat(src)).apply(null, vals);
    };

    return builder;
};
},{}],5:[function(require,module,exports){
'use strict';

var PATH_REPLACE_EXPR = /\[.+?\]/g,
    PATH_PROP_REPLACE_EXPR = /\[?(.*?)?\]/,
    REGEX_ESCAPE_EXPR = /[\/]/g,
    VALID_IDENTIFIER_EXPR = /^[a-z_$][0-9a-z]*$/gi,
    INVALID_SCHEMA = 'jsen: invalid schema object',
    browser = typeof window === 'object' && !!window.navigator,   // jshint ignore: line
    func = require('./func.js'),
    equal = require('./equal.js'),
    unique = require('./unique.js'),
    SchemaResolver = require('./resolver.js'),
    formats = require('./formats.js'),
    types = {},
    keywords = {};

function inlineRegex(regex) {
    var str = regex instanceof RegExp ? regex.toString() : new RegExp(regex).toString();

    if (browser) {
        return str;
    }

    str = str.substr(1, str.length - 2);
    str = '/' + str.replace(REGEX_ESCAPE_EXPR, '\\$&') + '/';

    return str;
}

function appendToPath(path, key) {
    VALID_IDENTIFIER_EXPR.lastIndex = 0;

    return VALID_IDENTIFIER_EXPR.test(key) ?
        path + '.' + key :
        path + '["' + key + '"]';
}

function type(obj) {
    var str = Object.prototype.toString.call(obj);
    return str.substr(8, str.length - 9).toLowerCase();
}

function isInteger(obj) {
    return (obj | 0) === obj;   // jshint ignore: line
}

types['null'] = function (path) {
    return path + ' === null';
};

types.boolean = function (path) {
    return 'typeof ' + path + ' === "boolean"';
};

types.string = function (path) {
    return 'typeof ' + path + ' === "string"';
};

types.number = function (path) {
    return 'typeof ' + path + ' === "number"';
};

types.integer = function (path) {
    return 'typeof ' + path + ' === "number" && !(' + path + ' % 1)';
};

types.array = function (path) {
    return path + ' !== undefined && Array.isArray(' + path + ')';
};

types.object = function (path) {
    return path + ' !== undefined && typeof ' + path + ' === "object" && ' + path + ' !== null && !Array.isArray(' + path + ')';
};

types.date = function (path) {
    return path + ' !== undefined && ' + path + ' instanceof Date';
};

keywords.type = function (context) {
    if (!context.schema.type) {
        return;
    }

    var specified = Array.isArray(context.schema.type) ? context.schema.type : [context.schema.type],
        src = specified.map(function mapType(type) {
            return types[type] ? types[type](context.path) || 'true' : 'true';
        }).join(' || ');

    if (src) {
        context.code('if (!(' + src + ')) {');

        context.error('type');

        context.code('}');
    }
};

keywords['enum'] = function (context) {
    var arr = context.schema['enum'],
        clauses = [],
        value, enumType, i;

    if (!Array.isArray(arr)) {
        return;
    }

    for (i = 0; i < arr.length; i++) {
        value = arr[i];
        enumType = typeof value;

        if (value === null || ['boolean', 'number', 'string'].indexOf(enumType) > -1) {
            // simple equality check for simple data types
            if (enumType === 'string') {
                clauses.push(context.path + ' === "' + value + '"');
            }
            else {
                clauses.push(context.path + ' === ' + value);
            }
        }
        else {
            // deep equality check for complex types or regexes
            clauses.push('equal(' + context.path + ', ' + JSON.stringify(value) + ')');
        }
    }

    context.code('if (!(' + clauses.join(' || ') + ')) {');
    context.error('enum');
    context.code('}');
};

keywords.minimum = function (context) {
    if (typeof context.schema.minimum === 'number') {
        context.code('if (' + context.path + ' < ' + context.schema.minimum + ') {');
        context.error('minimum');
        context.code('}');
    }
};

keywords.exclusiveMinimum = function (context) {
    if (context.schema.exclusiveMinimum === true && typeof context.schema.minimum === 'number') {
        context.code('if (' + context.path + ' === ' + context.schema.minimum + ') {');
        context.error('exclusiveMinimum');
        context.code('}');
    }
};

keywords.maximum = function (context) {
    if (typeof context.schema.maximum === 'number') {
        context.code('if (' + context.path + ' > ' + context.schema.maximum + ') {');
        context.error('maximum');
        context.code('}');
    }
};

keywords.exclusiveMaximum = function (context) {
    if (context.schema.exclusiveMaximum === true && typeof context.schema.maximum === 'number') {
        context.code('if (' + context.path + ' === ' + context.schema.maximum + ') {');
        context.error('exclusiveMaximum');
        context.code('}');
    }
};

keywords.multipleOf = function (context) {
    if (typeof context.schema.multipleOf === 'number') {
        var mul = context.schema.multipleOf,
            decimals = mul.toString().length - mul.toFixed(0).length - 1,
            pow = decimals > 0 ? Math.pow(10, decimals) : 1,
            path = context.path;

        if (decimals > 0) {
            context.code('if (+(Math.round((' + path + ' * ' + pow + ') + "e+" + ' + decimals + ') + "e-" + ' + decimals + ') % ' + (mul * pow) + ' !== 0) {');
        } else {
            context.code('if (((' + path + ' * ' + pow + ') % ' + (mul * pow) + ') !== 0) {');
        }

        context.error('multipleOf');
        context.code('}');
    }
};

keywords.minLength = function (context) {
    if (isInteger(context.schema.minLength)) {
        context.code('if (' + context.path + '.length < ' + context.schema.minLength + ') {');
        context.error('minLength');
        context.code('}');
    }
};

keywords.maxLength = function (context) {
    if (isInteger(context.schema.maxLength)) {
        context.code('if (' + context.path + '.length > ' + context.schema.maxLength + ') {');
        context.error('maxLength');
        context.code('}');
    }
};

keywords.pattern = function (context) {
    var regex = typeof context.schema.pattern === 'string' ?
        new RegExp(context.schema.pattern) :
        context.schema.pattern;

    if (type(regex) === 'regexp') {
        context.code('if (!(' + inlineRegex(regex) + ').test(' + context.path + ')) {');
        context.error('pattern');
        context.code('}');
    }
};

keywords.format = function (context) {
    if (typeof context.schema.format !== 'string' || !formats[context.schema.format]) {
        return;
    }

    context.code('if (!(' + formats[context.schema.format] + ').test(' + context.path + ')) {');
    context.error('format');
    context.code('}');
};

keywords.minItems = function (context) {
    if (isInteger(context.schema.minItems)) {
        context.code('if (' + context.path + '.length < ' + context.schema.minItems + ') {');
        context.error('minItems');
        context.code('}');
    }
};

keywords.maxItems = function (context) {
    if (isInteger(context.schema.maxItems)) {
        context.code('if (' + context.path + '.length > ' + context.schema.maxItems + ') {');
        context.error('maxItems');
        context.code('}');
    }
};

keywords.additionalItems = function (context) {
    if (context.schema.additionalItems === false && Array.isArray(context.schema.items)) {
        context.code('if (' + context.path + '.length > ' + context.schema.items.length + ') {');
        context.error('additionalItems');
        context.code('}');
    }
};

keywords.uniqueItems = function (context) {
    if (context.schema.uniqueItems) {
        context.code('if (unique(' + context.path + ').length !== ' + context.path + '.length) {');
        context.error('uniqueItems');
        context.code('}');
    }
};

keywords.items = function (context) {
    var index = context.declare(0),
        i = 0;

    if (type(context.schema.items) === 'object') {
        context.code('for (' + index + '; ' + index + ' < ' + context.path + '.length; ' + index + '++) {');

        context.validate(context.path + '[' + index + ']', context.schema.items, context.noFailFast);

        context.code('}');
    }
    else if (Array.isArray(context.schema.items)) {
        for (; i < context.schema.items.length; i++) {
            context.code('if (' + context.path + '.length - 1 >= ' + i + ') {');

            context.validate(context.path + '[' + i + ']', context.schema.items[i], context.noFailFast);

            context.code('}');
        }

        if (type(context.schema.additionalItems) === 'object') {
            context.code('for (' + index + ' = ' + i + '; ' + index + ' < ' + context.path + '.length; ' + index + '++) {');

            context.validate(context.path + '[' + index + ']', context.schema.additionalItems, context.noFailFast);

            context.code('}');
        }
    }
};

keywords.maxProperties = function (context) {
    if (isInteger(context.schema.maxProperties)) {
        context.code('if (Object.keys(' + context.path + ').length > ' + context.schema.maxProperties + ') {');
        context.error('maxProperties');
        context.code('}');
    }
};

keywords.minProperties = function (context) {
    if (isInteger(context.schema.minProperties)) {
        context.code('if (Object.keys(' + context.path + ').length < ' + context.schema.minProperties + ') {');
        context.error('minProperties');
        context.code('}');
    }
};

keywords.required = function (context) {
    if (!Array.isArray(context.schema.required)) {
        return;
    }

    for (var i = 0; i < context.schema.required.length; i++) {
        context.code('if (' + appendToPath(context.path, context.schema.required[i]) + ' === undefined) {');
        context.error('required', context.schema.required[i]);
        context.code('}');
    }
};

keywords.properties = function (context) {
    if (context.validatedProperties) {
        // prevent multiple generations of property validation
        return;
    }

    var props = context.schema.properties,
        propKeys = type(props) === 'object' ? Object.keys(props) : [],
        patProps = context.schema.patternProperties,
        patterns = type(patProps) === 'object' ? Object.keys(patProps) : [],
        addProps = context.schema.additionalProperties,
        addPropsCheck = addProps === false || type(addProps) === 'object',
        prop, i, nestedPath;

    // do not use this generator if we have patternProperties or additionalProperties
    // instead, the generator below will be used for all three keywords
    if (!propKeys.length || patterns.length || addPropsCheck) {
        return;
    }

    for (i = 0; i < propKeys.length; i++) {
        prop = propKeys[i];
        nestedPath = appendToPath(context.path, prop);

        context.code('if (' + nestedPath + ' !== undefined) {');

        context.validate(nestedPath, props[prop], context.noFailFast);

        context.code('}');
    }

    context.validatedProperties = true;
};

keywords.patternProperties = keywords.additionalProperties = function (context) {
    if (context.validatedProperties) {
        // prevent multiple generations of this function
        return;
    }

    var props = context.schema.properties,
        propKeys = type(props) === 'object' ? Object.keys(props) : [],
        patProps = context.schema.patternProperties,
        patterns = type(patProps) === 'object' ? Object.keys(patProps) : [],
        addProps = context.schema.additionalProperties,
        addPropsCheck = addProps === false || type(addProps) === 'object',
        keys, key, n, found,
        propKey, pattern, i;

    if (!propKeys.length && !patterns.length && !addPropsCheck) {
        return;
    }

    keys = context.declare('[]');
    key = context.declare('""');
    n = context.declare(0);

    if (addPropsCheck) {
        found = context.declare(false);
    }

    context.code(keys + ' = Object.keys(' + context.path + ')');

    context.code('for (' + n + '; ' + n + ' < ' + keys + '.length; ' + n + '++) {')
        (key + ' = ' + keys + '[' + n + ']')

        ('if (' + context.path + '[' + key + '] === undefined) {')
            ('continue')
        ('}');

    if (addPropsCheck) {
        context.code(found + ' = false');
    }

    // validate regular properties
    for (i = 0; i < propKeys.length; i++) {
        propKey = propKeys[i];

        context.code((i ? 'else ' : '') + 'if (' + key + ' === "' + propKey + '") {');

        if (addPropsCheck) {
            context.code(found + ' = true');
        }

        context.validate(appendToPath(context.path, propKey), props[propKey], context.noFailFast);

        context.code('}');
    }

    // validate pattern properties
    for (i = 0; i < patterns.length; i++) {
        pattern = patterns[i];

        context.code('if ((' + inlineRegex(pattern) + ').test(' + key + ')) {');

        if (addPropsCheck) {
            context.code(found + ' = true');
        }

        context.validate(context.path + '[' + key + ']', patProps[pattern], context.noFailFast);

        context.code('}');
    }

    // validate additional properties
    if (addPropsCheck) {
        context.code('if (!' + found + ') {');

        if (addProps === false) {
            // do not allow additional properties
            context.error('additionalProperties');
        }
        else {
            // validate additional properties
            context.validate(context.path + '[' + key + ']', addProps, context.noFailFast);
        }

        context.code('}');
    }

    context.code('}');

    context.validatedProperties = true;
};

keywords.dependencies = function (context) {
    if (type(context.schema.dependencies) !== 'object') {
        return;
    }

    var key, dep, i = 0;

    for (key in context.schema.dependencies) {
        dep = context.schema.dependencies[key];

        context.code('if (' + appendToPath(context.path, key) + ' !== undefined) {');

        if (type(dep) === 'object') {
            //schema dependency
            context.validate(context.path, dep, context.noFailFast);
        }
        else {
            // property dependency
            for (i; i < dep.length; i++) {
                context.code('if (' + appendToPath(context.path, dep[i]) + ' === undefined) {');
                context.error('dependencies', dep[i]);
                context.code('}');
            }
        }

        context.code('}');
    }
};

keywords.allOf = function (context) {
    if (!Array.isArray(context.schema.allOf)) {
        return;
    }

    for (var i = 0; i < context.schema.allOf.length; i++) {
        context.validate(context.path, context.schema.allOf[i], context.noFailFast);
    }
};

keywords.anyOf = function (context) {
    if (!Array.isArray(context.schema.anyOf)) {
        return;
    }

    var errCount = context.declare(0),
        initialCount = context.declare(0),
        found = context.declare(false),
        i = 0;

    context.code(initialCount + ' = errors.length');

    for (; i < context.schema.anyOf.length; i++) {
        context.code('if (!' + found + ') {');

        context.code(errCount + ' = errors.length');

        context.validate(context.path, context.schema.anyOf[i], true);

        context.code(found + ' = errors.length === ' + errCount)
        ('}');
    }

    context.code('if (!' + found + ') {');

    context.error('anyOf');

    context.code('} else {')
        ('errors.length = ' + initialCount)
    ('}');
};

keywords.oneOf = function (context) {
    if (!Array.isArray(context.schema.oneOf)) {
        return;
    }

    var matching = context.declare(0),
        initialCount = context.declare(0),
        errCount = context.declare(0),
        i = 0;

    context.code(initialCount + ' = errors.length');

    for (; i < context.schema.oneOf.length; i++) {
        context.code(errCount + ' = errors.length');

        context.validate(context.path, context.schema.oneOf[i], true);

        context.code('if (errors.length === ' + errCount + ') {')
            (matching + '++')
        ('}');
    }

    context.code('if (' + matching + ' !== 1) {');

    context.error('oneOf');

    context.code('} else {')
        ('errors.length = ' + initialCount)
    ('}');
};

keywords.not = function (context) {
    if (type(context.schema.not) !== 'object') {
        return;
    }

    var errCount = context.declare(0);

    context.code(errCount + ' = errors.length');

    context.validate(context.path, context.schema.not, true);

    context.code('if (errors.length === ' + errCount + ') {');

    context.error('not');

    context.code('} else {')
        ('errors.length = ' + errCount)
    ('}');
};

['minimum', 'exclusiveMinimum', 'maximum', 'exclusiveMaximum', 'multipleOf']
    .forEach(function (keyword) { keywords[keyword].type = 'number'; });

['minLength', 'maxLength', 'pattern', 'format']
    .forEach(function (keyword) { keywords[keyword].type = 'string'; });

['minItems', 'maxItems', 'additionalItems', 'uniqueItems', 'items']
    .forEach(function (keyword) { keywords[keyword].type = 'array'; });

['maxProperties', 'minProperties', 'required', 'properties', 'patternProperties', 'additionalProperties', 'dependencies']
    .forEach(function (keyword) { keywords[keyword].type = 'object'; });

function getGenerators(schema) {
    var keys = Object.keys(schema),
        start = [],
        perType = {},
        gen, i;

    for (i = 0; i < keys.length; i++) {
        gen = keywords[keys[i]];

        if (!gen) {
            continue;
        }

        if (gen.type) {
            if (!perType[gen.type]) {
                perType[gen.type] = [];
            }

            perType[gen.type].push(gen);
        }
        else {
            start.push(gen);
        }
    }

    return start.concat(Object.keys(perType).reduce(function (arr, key) {
        return arr.concat(perType[key]);
    }, []));
}

function replaceIndexedProperty(match) {
    var index = match.replace(PATH_PROP_REPLACE_EXPR, '$1');

    if (!isNaN(+index)) {
        // numeric index in array
        return '.' + index;
    }
    else if (index[0] === '"') {
        // string key for an object property
        return '[\\"' + index.substr(1, index.length - 2) + '\\"]';
    }

    // variable containing the actual key
    return '." + ' + index + ' + "';
}

function getPathExpression(path) {
    return '"' + path.replace(PATH_REPLACE_EXPR, replaceIndexedProperty).substr(5) + '"';
}

function clone(obj) {
    var cloned = obj,
        objType = type(obj),
        key, i;

    if (objType === 'object') {
        cloned = {};

        for (key in obj) {
            cloned[key] = clone(obj[key]);
        }
    }
    else if (objType === 'array') {
        cloned = [];

        for (i = 0; i < obj.length; i++) {
            cloned[i] = clone(obj[i]);
        }
    }
    else if (objType === 'regexp') {
        return new RegExp(obj);
    }
    else if (objType === 'date') {
        return new Date(obj.toJSON());
    }

    return cloned;
}

function build(schema, def, additional, resolver) {
    var defType, defValue, key, i;

    if (type(schema) !== 'object') {
        return def;
    }

    schema = resolver.resolve(schema);

    if (def === undefined && schema.hasOwnProperty('default')) {
        def = clone(schema['default']);
    }

    defType = type(def);

    if (defType === 'object' && type(schema.properties) === 'object') {
        for (key in schema.properties) {
            defValue = build(schema.properties[key], def[key], additional, resolver);

            if (defValue !== undefined) {
                def[key] = defValue;
            }
        }

        for (key in def) {
            if (!(key in schema.properties) &&
                (schema.additionalProperties === false ||
                (additional === false && !schema.additionalProperties))) {
                delete def[key];
            }
        }
    }
    else if (defType === 'array' && schema.items) {
        if (type(schema.items) === 'array') {
            for (i = 0; i < schema.items.length; i++) {
                defValue = build(schema.items[i], def[i], additional, resolver);

                if (defValue !== undefined || i < def.length) {
                    def[i] = defValue;
                }
            }
        }
        else if (def.length) {
            for (i = 0; i < def.length; i++) {
                def[i] = build(schema.items, def[i], additional, resolver);
            }
        }
    }

    return def;
}

function jsen(schema, options) {
    if (type(schema) !== 'object') {
        throw new Error(INVALID_SCHEMA);
    }

    options = options || {};

    var missing$Ref = options.missing$Ref || false,
        resolver = new SchemaResolver(schema, options.schemas, missing$Ref),
        counter = 0,
        id = function () { return 'i' + (counter++); },
        funcache = {},
        compiled,
        refs = {
            errors: []
        },
        scope = {
            equal: equal,
            unique: unique,
            refs: refs
        };

    function cache(schema) {
        var deref = resolver.resolve(schema),
            ref = schema.$ref,
            cached = funcache[ref],
            func;

        if (!cached) {
            cached = funcache[ref] = {
                key: id(),
                func: function (data) {
                    return func(data);
                }
            };

            func = compile(deref);

            Object.defineProperty(cached.func, 'errors', {
                get: function () {
                    return func.errors;
                }
            });

            refs[cached.key] = cached.func;
        }

        return 'refs.' + cached.key;
    }

    function compile(schema) {
        function declare(def) {
            var variname = id();

            code.def(variname, def);

            return variname;
        }

        function validate(path, schema, noFailFast) {
            var context,
                cachedRef,
                pathExp,
                index,
                lastType,
                format,
                gens,
                gen,
                i;

            function error(keyword, key) {
                var varid,
                    errorPath = path,
                    message = (key && schema.properties && schema.properties[key] && schema.properties[key].requiredMessage) ||
                        schema.invalidMessage;

                if (!message) {
                    message = key && schema.properties && schema.properties[key] && schema.properties[key].messages &&
                        schema.properties[key].messages[keyword] ||
                        schema.messages && schema.messages[keyword];
                }

                if (path.indexOf('[') > -1) {
                    // create error objects dynamically when path contains indexed property expressions
                    errorPath = getPathExpression(path);

                    if (key) {
                        errorPath = errorPath ? errorPath + ' + ".' + key + '"' : key;
                    }

                    code('errors.push({')
                        ('path: ' +  errorPath + ', ')
                        ('keyword: "' + keyword + '"' + (message ? ',' : ''));

                    if (message) {
                        code('message: "' + message + '"');
                    }

                    code('})');
                }
                else {
                    // generate faster code when no indexed properties in the path
                    varid = id();

                    errorPath = errorPath.substr(5);

                    if (key) {
                        errorPath = errorPath ? errorPath + '.' + key : key;
                    }

                    refs[varid] = {
                        path: errorPath,
                        keyword: keyword
                    };

                    if (message) {
                        refs[varid].message = message;
                    }

                    code('errors.push(refs.' + varid + ')');
                }

                if (!noFailFast && !options.greedy) {
                    code('return (validate.errors = errors) && false');
                }
            }

            if (schema.$ref !== undefined) {
                cachedRef = cache(schema);
                pathExp = getPathExpression(path);
                index = declare(0);

                code('if (!' + cachedRef + '(' + path + ')) {')
                    ('if (' + cachedRef + '.errors) {')
                        ('errors.push.apply(errors, ' + cachedRef + '.errors)')
                        ('for (' + index + ' = 0; ' + index + ' < ' + cachedRef + '.errors.length; ' + index + '++) {')
                            ('if (' + cachedRef + '.errors[' + index + '].path) {')
                                ('errors[errors.length - ' + cachedRef + '.errors.length + ' + index + '].path = ' + pathExp +
                                    ' + "." + ' + cachedRef + '.errors[' + index + '].path')
                            ('} else {')
                                ('errors[errors.length - ' + cachedRef + '.errors.length + ' + index + '].path = ' + pathExp)
                            ('}')
                        ('}')
                    ('}')
                ('}');

                return;
            }

            context = {
                path: path,
                schema: schema,
                code: code,
                declare: declare,
                validate: validate,
                error: error,
                noFailFast: noFailFast
            };

            gens = getGenerators(schema);

            for (i = 0; i < gens.length; i++) {
                gen = gens[i];

                if (gen.type && lastType !== gen.type) {
                    if (lastType) {
                        code('}');
                    }

                    lastType = gen.type;

                    code('if (' + types[gen.type](path) + ') {');
                }

                gen(context);
            }

            if (lastType) {
                code('}');
            }

            if (schema.format && options.formats) {
                format = options.formats[schema.format];

                if (format) {
                    if (typeof format === 'string' || format instanceof RegExp) {
                        code('if (!(' + inlineRegex(format) + ').test(' + context.path + ')) {');
                        error('format');
                        code('}');
                    }
                    else if (typeof format === 'function') {
                        (scope.formats || (scope.formats = {}))[schema.format] = format;
                        (scope.schemas || (scope.schemas = {}))[schema.format] = schema;

                        code('if (!formats["' + schema.format + '"](' + context.path + ', schemas["' + schema.format + '"])) {');
                        error('format');
                        code('}');
                    }
                }
            }
        }

        var code = func('validate', 'data')
            ('var errors = []');

        validate('data', schema);

        code('return (validate.errors = errors) && errors.length === 0');

        compiled = code.compile(scope);

        compiled.errors = [];

        compiled.build = function (initial, options) {
            return build(
                schema,
                (options && options.copy === false ? initial : clone(initial)),
                options && options.additionalProperties,
                resolver);
        };

        return compiled;
    }

    return compile(schema);
}

jsen.browser = browser;
jsen.clone = clone;
jsen.equal = equal;
jsen.unique = unique;

module.exports = jsen;

},{"./equal.js":2,"./formats.js":3,"./func.js":4,"./resolver.js":7,"./unique.js":8}],6:[function(require,module,exports){
module.exports={
    "id": "http://json-schema.org/draft-04/schema#",
    "$schema": "http://json-schema.org/draft-04/schema#",
    "description": "Core schema meta-schema",
    "definitions": {
        "schemaArray": {
            "type": "array",
            "minItems": 1,
            "items": { "$ref": "#" }
        },
        "positiveInteger": {
            "type": "integer",
            "minimum": 0
        },
        "positiveIntegerDefault0": {
            "allOf": [ { "$ref": "#/definitions/positiveInteger" }, { "default": 0 } ]
        },
        "simpleTypes": {
            "anyOf": [
                { "enum": [ "array", "boolean", "integer", "null", "number", "object", "string", "any" ] },
                { "type": "string" }
            ]
        },
        "stringArray": {
            "type": "array",
            "items": { "type": "string" },
            "minItems": 1,
            "uniqueItems": true
        }
    },
    "type": "object",
    "properties": {
        "id": {
            "type": "string",
            "format": "uri"
        },
        "$schema": {
            "type": "string",
            "format": "uri"
        },
        "title": {
            "type": "string"
        },
        "description": {
            "type": "string"
        },
        "default": {},
        "multipleOf": {
            "type": "number",
            "minimum": 0,
            "exclusiveMinimum": true
        },
        "maximum": {
            "type": "number"
        },
        "exclusiveMaximum": {
            "type": "boolean",
            "default": false
        },
        "minimum": {
            "type": "number"
        },
        "exclusiveMinimum": {
            "type": "boolean",
            "default": false
        },
        "maxLength": { "$ref": "#/definitions/positiveInteger" },
        "minLength": { "$ref": "#/definitions/positiveIntegerDefault0" },
        "pattern": {
            "type": "string",
            "format": "regex"
        },
        "additionalItems": {
            "anyOf": [
                { "type": "boolean" },
                { "$ref": "#" }
            ],
            "default": {}
        },
        "items": {
            "anyOf": [
                { "$ref": "#" },
                { "$ref": "#/definitions/schemaArray" }
            ],
            "default": {}
        },
        "maxItems": { "$ref": "#/definitions/positiveInteger" },
        "minItems": { "$ref": "#/definitions/positiveIntegerDefault0" },
        "uniqueItems": {
            "type": "boolean",
            "default": false
        },
        "maxProperties": { "$ref": "#/definitions/positiveInteger" },
        "minProperties": { "$ref": "#/definitions/positiveIntegerDefault0" },
        "required": { "$ref": "#/definitions/stringArray" },
        "additionalProperties": {
            "anyOf": [
                { "type": "boolean" },
                { "$ref": "#" }
            ],
            "default": {}
        },
        "definitions": {
            "type": "object",
            "additionalProperties": { "$ref": "#" },
            "default": {}
        },
        "properties": {
            "type": "object",
            "additionalProperties": { "$ref": "#" },
            "default": {}
        },
        "patternProperties": {
            "type": "object",
            "additionalProperties": { "$ref": "#" },
            "default": {}
        },
        "dependencies": {
            "type": "object",
            "additionalProperties": {
                "anyOf": [
                    { "$ref": "#" },
                    { "$ref": "#/definitions/stringArray" }
                ]
            }
        },
        "enum": {
            "type": "array",
            "minItems": 1,
            "uniqueItems": true
        },
        "type": {
            "anyOf": [
                { "$ref": "#/definitions/simpleTypes" },
                {
                    "type": "array",
                    "items": { "$ref": "#/definitions/simpleTypes" },
                    "minItems": 1,
                    "uniqueItems": true
                }
            ]
        },
        "allOf": { "$ref": "#/definitions/schemaArray" },
        "anyOf": { "$ref": "#/definitions/schemaArray" },
        "oneOf": { "$ref": "#/definitions/schemaArray" },
        "not": { "$ref": "#" }
    },
    "dependencies": {
        "exclusiveMaximum": [ "maximum" ],
        "exclusiveMinimum": [ "minimum" ]
    },
    "default": {}
}

},{}],7:[function(require,module,exports){
'use strict';

var metaschema = require('./metaschema.json'),
    refRegex = /#?(\/?\w+)*$/,
    INVALID_SCHEMA_REFERENCE = 'jsen: invalid schema reference';

function get(obj, key) {
    var parts = key.split('.'),
        subobj,
        remaining;

    if (parts.length === 1) {
        // simple key
        return obj[key];
    }

    // compound and nested properties
    // e.g. key('nested.key', { nested: { key: 123 } }) === 123
    // e.g. key('compount.key', { 'compound.key': 456 }) === 456
    while (parts.length && obj !== undefined && obj !== null) {
        // take a part from the front
        remaining = parts.slice(0);
        subobj = undefined;

        // try to match larger compound keys containing dots
        while (remaining.length && subobj === undefined) {
            subobj = obj[remaining.join('.')];

            if (subobj === undefined) {
                remaining.pop();
            }
        }

        // if there is a matching larger compount key, use that
        if (subobj !== undefined) {
            obj = subobj;

            // remove keys from the parts, respectively
            while (remaining.length) {
                remaining.shift();
                parts.shift();
            }
        }
        else {
            // treat like normal simple keys
            obj = obj[parts.shift()];
        }
    }

    return obj;
}

// http://tools.ietf.org/html/draft-ietf-appsawg-json-pointer-08#section-3
function unescape(pointer) {
    return decodeURIComponent(pointer)
        .replace(/~1/g, '/')
        .replace(/~0/g, '~');
}

function refToPath(ref) {
    if (ref.indexOf('#') < 0) {
        return ref;
    }

    var path = ref.split('#')[1];

    if (path) {
        path = path
            .split('/')
            .map(unescape)
            .join('.');

        if (path[0] === '.') {
            path = path.substr(1);
        }
    }

    return path;
}

function refFromId(obj, ref) {
    if (obj && typeof obj === 'object') {
        if (obj.id === ref) {
            return obj;
        }

        return Object.keys(obj).reduce(function (resolved, key) {
            return resolved || refFromId(obj[key], ref);
        }, undefined);
    }

    return undefined;
}

function getResolvers(schemas) {
    var keys = Object.keys(schemas),
        resolvers = {},
        key, i;

    for (i = 0; i < keys.length; i++) {
        key = keys[i];
        resolvers[key] = new SchemaResolver(schemas[key]);
    }

    return resolvers;
}

function SchemaResolver(rootSchema, external, missing$Ref) {  // jshint ignore: line
    this.rootSchema = rootSchema;
    this.cache = {};
    this.resolved = null;
    this.missing$Ref = missing$Ref;

    this.resolvers = external && typeof external === 'object' ?
        getResolvers(external) :
        null;
}

SchemaResolver.prototype.resolveRef = function (ref) {
    var err = new Error(INVALID_SCHEMA_REFERENCE + ' ' + ref),
        root = this.rootSchema,
        externalResolver,
        path,
        dest;

    if (!ref || typeof ref !== 'string' || !refRegex.test(ref)) {
        throw err;
    }

    if (ref === metaschema.id) {
        dest = metaschema;
    }

    if (!dest) {
        dest = refFromId(root, ref);
    }

    if (!dest) {
        path = refToPath(ref);

        dest = path ? get(root, path) : root;
    }

    if (!dest && path && this.resolvers) {
        externalResolver = get(this.resolvers, path);

        if (externalResolver) {
            dest = externalResolver.resolve(externalResolver.rootSchema);
        }
    }

    if (!dest || typeof dest !== 'object') {
        if (this.missing$Ref) {
            dest = {};
        } else {
            throw err;
        }
    }

    if (this.cache[ref] === dest) {
        return dest;
    }

    this.cache[ref] = dest;

    if (dest.$ref !== undefined) {
        dest = this.cache[ref] = this.resolveRef(dest.$ref);
    }

    return dest;
};

SchemaResolver.prototype.resolve = function (schema) {
    if (!schema || typeof schema !== 'object') {
        return schema;
    }

    var ref = schema.$ref,
        resolved = this.cache[ref];

    if (ref === undefined) {
        return schema;
    }

    if (resolved) {
        return resolved;
    }

    resolved = this.resolveRef(ref);

    if (schema === this.rootSchema && schema !== resolved) {
        // substitute the resolved root schema
        this.rootSchema = resolved;
    }

    return resolved;
};

module.exports = SchemaResolver;
},{"./metaschema.json":6}],8:[function(require,module,exports){
'use strict';

var equal = require('./equal.js');

function findIndex(arr, value, comparator) {
    for (var i = 0, len = arr.length; i < len; i++) {
        if (comparator(arr[i], value)) {
            return i;
        }
    }

    return -1;
}

module.exports = function unique(arr) {
    return arr.filter(function uniqueOnly(value, index, self) {
        return findIndex(self, value, equal) === index;
    });
};

module.exports.findIndex = findIndex;
},{"./equal.js":2}]},{},[1])(1)
});
	
/**
 * @license MIT
 * @fileOverview Favico animations
 * @author Miroslav Magda, http://blog.ejci.net
 * @version 0.3.9
 */

/**
 * Create new favico instance
 * @param {Object} Options
 * @return {Object} Favico object
 * @example
 * var favico = new Favico({
 *    bgColor : '#d00',
 *    textColor : '#fff',
 *    fontFamily : 'sans-serif',
 *    fontStyle : 'bold',
 *    position : 'down',
 *    type : 'circle',
 *    animation : 'slide',
 *    dataUrl: function(url){},
 *    win: top
 * });
 */
(function() {

	var Favico = (function(opt) {
		'use strict';
		opt = (opt) ? opt : {};
		var _def = {
			bgColor : '#d00',
			textColor : '#fff',
			fontFamily : 'sans-serif', //Arial,Verdana,Times New Roman,serif,sans-serif,...
			fontStyle : 'bold', //normal,italic,oblique,bold,bolder,lighter,100,200,300,400,500,600,700,800,900
			type : 'circle',
			position : 'down', // down, up, left, leftup (upleft)
			animation : 'slide',
			elementId : false,
			dataUrl : false,
			win: window
		};
		var _opt, _orig, _h, _w, _canvas, _context, _img, _ready, _lastBadge, _running, _readyCb, _stop, _browser, _animTimeout, _drawTimeout, _doc;

		_browser = {};
		_browser.ff = typeof InstallTrigger != 'undefined';
		_browser.chrome = !!window.chrome;
		_browser.opera = !!window.opera || navigator.userAgent.indexOf('Opera') >= 0;
		_browser.ie = /*@cc_on!@*/false;
		_browser.safari = Object.prototype.toString.call(window.HTMLElement).indexOf('Constructor') > 0;
		_browser.supported = (_browser.chrome || _browser.ff || _browser.opera);

		var _queue = [];
		_readyCb = function() {
		};
		_ready = _stop = false;
		/**
		 * Initialize favico
		 */
		var init = function() {
			//merge initial options
			_opt = merge(_def, opt);
			_opt.bgColor = hexToRgb(_opt.bgColor);
			_opt.textColor = hexToRgb(_opt.textColor);
			_opt.position = _opt.position.toLowerCase();
			_opt.animation = (animation.types['' + _opt.animation]) ? _opt.animation : _def.animation;

			_doc = _opt.win.document;

			var isUp = _opt.position.indexOf('up') > -1;
			var isLeft = _opt.position.indexOf('left') > -1;

			//transform animation
			if (isUp || isLeft) {
				for (var i = 0; i < animation.types['' + _opt.animation].length; i++) {
					var step = animation.types['' + _opt.animation][i];

					if (isUp) {
						if (step.y < 0.6) {
							step.y = step.y - 0.4;
						} else {
							step.y = step.y - 2 * step.y + (1 - step.w);
						}
					}

					if (isLeft) {
						if (step.x < 0.6) {
							step.x = step.x - 0.4;
						} else {
							step.x = step.x - 2 * step.x + (1 - step.h);
						}
					}

					animation.types['' + _opt.animation][i] = step;
				}
			}
			_opt.type = (type['' + _opt.type]) ? _opt.type : _def.type;

			_orig = link.getIcon();
			//create temp canvas
			_canvas = document.createElement('canvas');
			//create temp image
			_img = document.createElement('img');
			if (_orig.hasAttribute('href')) {
				_img.setAttribute('crossOrigin', 'anonymous');
				_img.setAttribute('src', _orig.getAttribute('href'));
				//get width/height
				_img.onload = function() {
					_h = (_img.height > 0) ? _img.height : 32;
					_w = (_img.width > 0) ? _img.width : 32;
					_canvas.height = _h;
					_canvas.width = _w;
					_context = _canvas.getContext('2d');
					icon.ready();
				};
			} else {
				_img.setAttribute('src', '');
				_h = 32;
				_w = 32;
				_img.height = _h;
				_img.width = _w;
				_canvas.height = _h;
				_canvas.width = _w;
				_context = _canvas.getContext('2d');
				icon.ready();
			}

		};
		/**
		 * Icon namespace
		 */
		var icon = {};
		/**
		 * Icon is ready (reset icon) and start animation (if ther is any)
		 */
		icon.ready = function() {
			_ready = true;
			icon.reset();
			_readyCb();
		};
		/**
		 * Reset icon to default state
		 */
		icon.reset = function() {
			//reset
			if (!_ready) {
				return;
			}
			_queue = [];
			_lastBadge = false;
			_running = false;
			_context.clearRect(0, 0, _w, _h);
			_context.drawImage(_img, 0, 0, _w, _h);
			//_stop=true;
			link.setIcon(_canvas);
			//webcam('stop');
			//video('stop');
			window.clearTimeout(_animTimeout);
			window.clearTimeout(_drawTimeout);
		};
		/**
		 * Start animation
		 */
		icon.start = function() {
			if (!_ready || _running) {
				return;
			}
			var finished = function() {
				_lastBadge = _queue[0];
				_running = false;
				if (_queue.length > 0) {
					_queue.shift();
					icon.start();
				} else {

				}
			};
			if (_queue.length > 0) {
				_running = true;
				var run = function() {
					// apply options for this animation
					['type', 'animation', 'bgColor', 'textColor', 'fontFamily', 'fontStyle'].forEach(function(a) {
						if ( a in _queue[0].options) {
							_opt[a] = _queue[0].options[a];
						}
					});
					animation.run(_queue[0].options, function() {
						finished();
					}, false);
				};
				if (_lastBadge) {
					animation.run(_lastBadge.options, function() {
						run();
					}, true);
				} else {
					run();
				}
			}
		};

		/**
		 * Badge types
		 */
		var type = {};
		var options = function(opt) {
			opt.n = (( typeof opt.n) === 'number') ? Math.abs(opt.n | 0) : opt.n;
			opt.x = _w * opt.x;
			opt.y = _h * opt.y;
			opt.w = _w * opt.w;
			opt.h = _h * opt.h;
			opt.len = ("" + opt.n).length;
			return opt;
		};
		/**
		 * Generate circle
		 * @param {Object} opt Badge options
		 */
		type.circle = function(opt) {
			opt = options(opt);
			var more = false;
			if (opt.len === 2) {
				opt.x = opt.x - opt.w * 0.4;
				opt.w = opt.w * 1.4;
				more = true;
			} else if (opt.len >= 3) {
				opt.x = opt.x - opt.w * 0.65;
				opt.w = opt.w * 1.65;
				more = true;
			}
			_context.clearRect(0, 0, _w, _h);
			_context.drawImage(_img, 0, 0, _w, _h);
			_context.beginPath();
			_context.font = _opt.fontStyle + " " + Math.floor(opt.h * (opt.n > 99 ? 0.85 : 1)) + "px " + _opt.fontFamily;
			_context.textAlign = 'center';
			if (more) {
				_context.moveTo(opt.x + opt.w / 2, opt.y);
				_context.lineTo(opt.x + opt.w - opt.h / 2, opt.y);
				_context.quadraticCurveTo(opt.x + opt.w, opt.y, opt.x + opt.w, opt.y + opt.h / 2);
				_context.lineTo(opt.x + opt.w, opt.y + opt.h - opt.h / 2);
				_context.quadraticCurveTo(opt.x + opt.w, opt.y + opt.h, opt.x + opt.w - opt.h / 2, opt.y + opt.h);
				_context.lineTo(opt.x + opt.h / 2, opt.y + opt.h);
				_context.quadraticCurveTo(opt.x, opt.y + opt.h, opt.x, opt.y + opt.h - opt.h / 2);
				_context.lineTo(opt.x, opt.y + opt.h / 2);
				_context.quadraticCurveTo(opt.x, opt.y, opt.x + opt.h / 2, opt.y);
			} else {
				_context.arc(opt.x + opt.w / 2, opt.y + opt.h / 2, opt.h / 2, 0, 2 * Math.PI);
			}
			_context.fillStyle = 'rgba(' + _opt.bgColor.r + ',' + _opt.bgColor.g + ',' + _opt.bgColor.b + ',' + opt.o + ')';
			_context.fill();
			_context.closePath();
			_context.beginPath();
			_context.stroke();
			_context.fillStyle = 'rgba(' + _opt.textColor.r + ',' + _opt.textColor.g + ',' + _opt.textColor.b + ',' + opt.o + ')';
			//_context.fillText((more) ? '9+' : opt.n, Math.floor(opt.x + opt.w / 2), Math.floor(opt.y + opt.h - opt.h * 0.15));
			if (( typeof opt.n) === 'number' && opt.n > 999) {
				_context.fillText(((opt.n > 9999) ? 9 : Math.floor(opt.n / 1000) ) + 'k+', Math.floor(opt.x + opt.w / 2), Math.floor(opt.y + opt.h - opt.h * 0.2));
			} else {
				_context.fillText(opt.n, Math.floor(opt.x + opt.w / 2), Math.floor(opt.y + opt.h - opt.h * 0.15));
			}
			_context.closePath();
		};
		/**
		 * Generate rectangle
		 * @param {Object} opt Badge options
		 */
		type.rectangle = function(opt) {
			opt = options(opt);
			var more = false;
			if (opt.len === 2) {
				opt.x = opt.x - opt.w * 0.4;
				opt.w = opt.w * 1.4;
				more = true;
			} else if (opt.len >= 3) {
				opt.x = opt.x - opt.w * 0.65;
				opt.w = opt.w * 1.65;
				more = true;
			}
			_context.clearRect(0, 0, _w, _h);
			_context.drawImage(_img, 0, 0, _w, _h);
			_context.beginPath();
			_context.font = _opt.fontStyle + " " + Math.floor(opt.h * (opt.n > 99 ? 0.9 : 1)) + "px " + _opt.fontFamily;
			_context.textAlign = 'center';
			_context.fillStyle = 'rgba(' + _opt.bgColor.r + ',' + _opt.bgColor.g + ',' + _opt.bgColor.b + ',' + opt.o + ')';
			_context.fillRect(opt.x, opt.y, opt.w, opt.h);
			_context.fillStyle = 'rgba(' + _opt.textColor.r + ',' + _opt.textColor.g + ',' + _opt.textColor.b + ',' + opt.o + ')';
			//_context.fillText((more) ? '9+' : opt.n, Math.floor(opt.x + opt.w / 2), Math.floor(opt.y + opt.h - opt.h * 0.15));
			if (( typeof opt.n) === 'number' && opt.n > 999) {
				_context.fillText(((opt.n > 9999) ? 9 : Math.floor(opt.n / 1000) ) + 'k+', Math.floor(opt.x + opt.w / 2), Math.floor(opt.y + opt.h - opt.h * 0.2));
			} else {
				_context.fillText(opt.n, Math.floor(opt.x + opt.w / 2), Math.floor(opt.y + opt.h - opt.h * 0.15));
			}
			_context.closePath();
		};

		/**
		 * Set badge
		 */
		var badge = function(number, opts) {
			opts = (( typeof opts) === 'string' ? {
				animation : opts
			} : opts) || {};
			_readyCb = function() {
				try {
					if ( typeof (number) === 'number' ? (number > 0) : (number !== '')) {
						var q = {
							type : 'badge',
							options : {
								n : number
							}
						};
						if ('animation' in opts && animation.types['' + opts.animation]) {
							q.options.animation = '' + opts.animation;
						}
						if ('type' in opts && type['' + opts.type]) {
							q.options.type = '' + opts.type;
						}
						['bgColor', 'textColor'].forEach(function(o) {
							if ( o in opts) {
								q.options[o] = hexToRgb(opts[o]);
							}
						});
						['fontStyle', 'fontFamily'].forEach(function(o) {
							if ( o in opts) {
								q.options[o] = opts[o];
							}
						});
						_queue.push(q);
						if (_queue.length > 100) {
							throw new Error('Too many badges requests in queue.');
						}
						icon.start();
					} else {
						icon.reset();
					}
				} catch(e) {
					throw new Error('Error setting badge. Message: ' + e.message);
				}
			};
			if (_ready) {
				_readyCb();
			}
		};

		/**
		 * Set image as icon
		 */
		var image = function(imageElement) {
			_readyCb = function() {
				try {
					var w = imageElement.width;
					var h = imageElement.height;
					var newImg = document.createElement('img');
					var ratio = (w / _w < h / _h) ? (w / _w) : (h / _h);
					newImg.setAttribute('crossOrigin', 'anonymous');
					newImg.setAttribute('src', imageElement.getAttribute('src'));
					newImg.height = (h / ratio);
					newImg.width = (w / ratio);
					_context.clearRect(0, 0, _w, _h);
					_context.drawImage(newImg, 0, 0, _w, _h);
					link.setIcon(_canvas);
				} catch(e) {
					throw new Error('Error setting image. Message: ' + e.message);
				}
			};
			if (_ready) {
				_readyCb();
			}
		};
		/**
		 * Set video as icon
		 */
		var video = function(videoElement) {
			_readyCb = function() {
				try {
					if (videoElement === 'stop') {
						_stop = true;
						icon.reset();
						_stop = false;
						return;
					}
					//var w = videoElement.width;
					//var h = videoElement.height;
					//var ratio = (w / _w < h / _h) ? (w / _w) : (h / _h);
					videoElement.addEventListener('play', function() {
						drawVideo(this);
					}, false);

				} catch(e) {
					throw new Error('Error setting video. Message: ' + e.message);
				}
			};
			if (_ready) {
				_readyCb();
			}
		};
		/**
		 * Set video as icon
		 */
		var webcam = function(action) {
			//UR
			if (!window.URL || !window.URL.createObjectURL) {
				window.URL = window.URL || {};
				window.URL.createObjectURL = function(obj) {
					return obj;
				};
			}
			if (_browser.supported) {
				var newVideo = false;
				navigator.getUserMedia = navigator.getUserMedia || navigator.oGetUserMedia || navigator.msGetUserMedia || navigator.mozGetUserMedia || navigator.webkitGetUserMedia;
				_readyCb = function() {
					try {
						if (action === 'stop') {
							_stop = true;
							icon.reset();
							_stop = false;
							return;
						}
						newVideo = document.createElement('video');
						newVideo.width = _w;
						newVideo.height = _h;
						navigator.getUserMedia({
							video : true,
							audio : false
						}, function(stream) {
							newVideo.src = URL.createObjectURL(stream);
							newVideo.play();
							drawVideo(newVideo);
						}, function() {
						});
					} catch(e) {
						throw new Error('Error setting webcam. Message: ' + e.message);
					}
				};
				if (_ready) {
					_readyCb();
				}
			}

		};

		/**
		 * Draw video to context and repeat :)
		 */
		function drawVideo(video) {
			if (video.paused || video.ended || _stop) {
				return false;
			}
			//nasty hack for FF webcam (Thanks to Julian Ćwirko, kontakt@redsunmedia.pl)
			try {
				_context.clearRect(0, 0, _w, _h);
				_context.drawImage(video, 0, 0, _w, _h);
			} catch(e) {

			}
			_drawTimeout = setTimeout(function() {
				drawVideo(video);
			}, animation.duration);
			link.setIcon(_canvas);
		}

		var link = {};
		/**
		 * Get icon from HEAD tag or create a new <link> element
		 */
		link.getIcon = function() {
			var elm = false;
			//get link element
			var getLink = function() {
				var link = _doc.getElementsByTagName('head')[0].getElementsByTagName('link');
				for (var l = link.length, i = (l - 1); i >= 0; i--) {
					if ((/(^|\s)icon(\s|$)/i).test(link[i].getAttribute('rel'))) {
						return link[i];
					}
				}
				return false;
			};
			if (_opt.element) {
				elm = _opt.element;
			} else if (_opt.elementId) {
				//if img element identified by elementId
				elm = _doc.getElementById(_opt.elementId);
				elm.setAttribute('href', elm.getAttribute('src'));
			} else {
				//if link element
				elm = getLink();
				if (elm === false) {
					elm = _doc.createElement('link');
					elm.setAttribute('rel', 'icon');
					_doc.getElementsByTagName('head')[0].appendChild(elm);
				}
			}
			elm.setAttribute('type', 'image/png');
			return elm;
		};
		link.setIcon = function(canvas) {
			var url = canvas.toDataURL('image/png');
			if (_opt.dataUrl) {
				//if using custom exporter
				_opt.dataUrl(url);
			}
			if (_opt.element) {
				_opt.element.setAttribute('href', url);
				_opt.element.setAttribute('src', url);
			} else if (_opt.elementId) {
				//if is attached to element (image)
				var elm = _doc.getElementById(_opt.elementId);
				elm.setAttribute('href', url);
				elm.setAttribute('src', url);
			} else {
				//if is attached to fav icon
				if (_browser.ff || _browser.opera) {
					//for FF we need to "recreate" element, atach to dom and remove old <link>
					//var originalType = _orig.getAttribute('rel');
					var old = _orig;
					_orig = _doc.createElement('link');
					//_orig.setAttribute('rel', originalType);
					if (_browser.opera) {
						_orig.setAttribute('rel', 'icon');
					}
					_orig.setAttribute('rel', 'icon');
					_orig.setAttribute('type', 'image/png');
					_doc.getElementsByTagName('head')[0].appendChild(_orig);
					_orig.setAttribute('href', url);
					if (old.parentNode) {
						old.parentNode.removeChild(old);
					}
				} else {
					_orig.setAttribute('href', url);
				}
			}
		};

		//http://stackoverflow.com/questions/5623838/rgb-to-hex-and-hex-to-rgb#answer-5624139
		//HEX to RGB convertor
		function hexToRgb(hex) {
			var shorthandRegex = /^#?([a-f\d])([a-f\d])([a-f\d])$/i;
			hex = hex.replace(shorthandRegex, function(m, r, g, b) {
				return r + r + g + g + b + b;
			});
			var result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
			return result ? {
				r : parseInt(result[1], 16),
				g : parseInt(result[2], 16),
				b : parseInt(result[3], 16)
			} : false;
		}

		/**
		 * Merge options
		 */
		function merge(def, opt) {
			var mergedOpt = {};
			var attrname;
			for (attrname in def) {
				mergedOpt[attrname] = def[attrname];
			}
			for (attrname in opt) {
				mergedOpt[attrname] = opt[attrname];
			}
			return mergedOpt;
		}

		/**
		 * Cross-browser page visibility shim
		 * http://stackoverflow.com/questions/12536562/detect-whether-a-window-is-visible
		 */
		function isPageHidden() {
			return _doc.hidden || _doc.msHidden || _doc.webkitHidden || _doc.mozHidden;
		}

		/**
		 * @namespace animation
		 */
		var animation = {};
		/**
		 * Animation "frame" duration
		 */
		animation.duration = 40;
		/**
		 * Animation types (none,fade,pop,slide)
		 */
		animation.types = {};
		animation.types.fade = [{
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 0.0
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 0.1
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 0.2
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 0.3
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 0.4
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 0.5
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 0.6
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 0.7
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 0.8
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 0.9
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 1.0
		}];
		animation.types.none = [{
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 1
		}];
		animation.types.pop = [{
			x : 1,
			y : 1,
			w : 0,
			h : 0,
			o : 1
		}, {
			x : 0.9,
			y : 0.9,
			w : 0.1,
			h : 0.1,
			o : 1
		}, {
			x : 0.8,
			y : 0.8,
			w : 0.2,
			h : 0.2,
			o : 1
		}, {
			x : 0.7,
			y : 0.7,
			w : 0.3,
			h : 0.3,
			o : 1
		}, {
			x : 0.6,
			y : 0.6,
			w : 0.4,
			h : 0.4,
			o : 1
		}, {
			x : 0.5,
			y : 0.5,
			w : 0.5,
			h : 0.5,
			o : 1
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 1
		}];
		animation.types.popFade = [{
			x : 0.75,
			y : 0.75,
			w : 0,
			h : 0,
			o : 0
		}, {
			x : 0.65,
			y : 0.65,
			w : 0.1,
			h : 0.1,
			o : 0.2
		}, {
			x : 0.6,
			y : 0.6,
			w : 0.2,
			h : 0.2,
			o : 0.4
		}, {
			x : 0.55,
			y : 0.55,
			w : 0.3,
			h : 0.3,
			o : 0.6
		}, {
			x : 0.50,
			y : 0.50,
			w : 0.4,
			h : 0.4,
			o : 0.8
		}, {
			x : 0.45,
			y : 0.45,
			w : 0.5,
			h : 0.5,
			o : 0.9
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 1
		}];
		animation.types.slide = [{
			x : 0.4,
			y : 1,
			w : 0.6,
			h : 0.6,
			o : 1
		}, {
			x : 0.4,
			y : 0.9,
			w : 0.6,
			h : 0.6,
			o : 1
		}, {
			x : 0.4,
			y : 0.9,
			w : 0.6,
			h : 0.6,
			o : 1
		}, {
			x : 0.4,
			y : 0.8,
			w : 0.6,
			h : 0.6,
			o : 1
		}, {
			x : 0.4,
			y : 0.7,
			w : 0.6,
			h : 0.6,
			o : 1
		}, {
			x : 0.4,
			y : 0.6,
			w : 0.6,
			h : 0.6,
			o : 1
		}, {
			x : 0.4,
			y : 0.5,
			w : 0.6,
			h : 0.6,
			o : 1
		}, {
			x : 0.4,
			y : 0.4,
			w : 0.6,
			h : 0.6,
			o : 1
		}];
		/**
		 * Run animation
		 * @param {Object} opt Animation options
		 * @param {Object} cb Callabak after all steps are done
		 * @param {Object} revert Reverse order? true|false
		 * @param {Object} step Optional step number (frame bumber)
		 */
		animation.run = function(opt, cb, revert, step) {
			var animationType = animation.types[isPageHidden() ? 'none' : _opt.animation];
			if (revert === true) {
				step = ( typeof step !== 'undefined') ? step : animationType.length - 1;
			} else {
				step = ( typeof step !== 'undefined') ? step : 0;
			}
			cb = (cb) ? cb : function() {
			};
			if ((step < animationType.length) && (step >= 0)) {
				type[_opt.type](merge(opt, animationType[step]));
				_animTimeout = setTimeout(function() {
					if (revert) {
						step = step - 1;
					} else {
						step = step + 1;
					}
					animation.run(opt, cb, revert, step);
				}, animation.duration);

				link.setIcon(_canvas);
			} else {
				cb();
				return;
			}
		};
		//auto init
		init();
		return {
			badge : badge,
			video : video,
			image : image,
			webcam : webcam,
			reset : icon.reset,
			browser : {
				supported : _browser.supported
			}
		};
	});

	// AMD / RequireJS
	if ( typeof define !== 'undefined' && define.amd) {
		define([], function() {
			return Favico;
		});
	}
	// CommonJS
	else if ( typeof module !== 'undefined' && module.exports) {
		module.exports = Favico;
	}
	// included directly via <script> tag
	else {
		window.Favico = Favico;
	}

})();
}

})();