- // ==UserScript==
- // @name pixiv 検索オプションを追加
- // @name:ja pixiv 検索オプションを追加
- // @name:en pixiv Search Options
- // @description This is a search option add-on script for pixiv. Enables you to select search mode by radio buttons (legacy feature).
- // @description:ja 検索オプションを、以前のようにラジオボタンで選択出来るようにする。
- // @namespace http://loda.jp/script/
- // @version 6.0.1
- // @match https://www.pixiv.net/*
- // @exclude https://www.pixiv.net/member_illust.php?*mode=manga*
- // @exclude https://www.pixiv.net/apps.php*
- // @require https://gitcdn.xyz/cdn/greasemonkey/gm4-polyfill/a834d46afcc7d6f6297829876423f58bb14a0d97/gm4-polyfill.js
- // @resource dialog-polyfill.css https://bowercdn.net/c/dialog-polyfill-0.4.10/dialog-polyfill.css
- // @require https://bowercdn.net/c/dialog-polyfill-0.4.10/dialog-polyfill.js
- // @require https://greasyfork.org/scripts/17895/code/polyfill.js?version=625392
- // @require https://greasyfork.org/scripts/19616/code/utilities.js?version=752462
- // @require https://greasyfork.org/scripts/17896/code/start-script.js?version=112958
- // @license MPL-2.0
- // @contributionURL https://www.amazon.co.jp/registry/wishlist/E7PJ5C3K7AM2
- // @compatible Edge 非推奨 / Deprecated
- // @compatible Firefox
- // @compatible Opera
- // @compatible Chrome
- // @grant GM.getValue
- // @grant GM_getValue
- // @grant GM.setValue
- // @grant GM_setValue
- // @grant GM.registerMenuCommand
- // @grant GM_registerMenuCommand
- // @grant GM.getResourceUrl
- // @grant GM_getResourceURL
- // @noframes
- // @run-at document-start
- // @icon 
- // @author 100の人
- // @homepageURL https://greasyfork.org/scripts/265
- // ==/UserScript==
-
- // 当スクリプトはpixivが作成、配布しているアプリケーションではありません。
- // <https://www.pixiv.net/terms/?page=brand>
-
- 'use strict';
-
- // L10N
- Gettext.setLocalizedTexts({
- /*eslint-disable quote-props, max-len */
- 'en': {
- '完全一致': 'Exact match',
- '部分一致': 'Partial match',
- 'タイトル・キャプション': 'Title/Description',
- '小説': 'Novels',
- 'タグ': 'Tags',
- 'キーワード': 'Keyword',
- '本文': 'Content',
- 'ユーザー': 'User',
- 'グループ': 'Groups',
- 'すべて': 'All',
- '検索': 'Search',
- '小説検索': 'Search novel',
- 'ユーザー検索': 'Search user',
- 'グループ検索': 'Search group',
- 'pixiv 検索オプションを追加': 'pixiv Search Options',
- 'この設定をオンにすると、ヘッダの高さが大きくなります。': 'If you set this, the page header becomes higher.',
- '別のタブでpixivを開いていた場合、この設定の変更はそのタブの再読み込み後に反映されます。': 'If you have opened pixiv pages in other tabs, this setting will reflect after the tab refresh.',
- 'イラスト・マンガの検索結果ページで、検索モードを維持': 'In search result pages of illusts and mangas, maintains current search mode',
- 'この設定をオフにすると、最初は「部分一致」にチェックが入った状態になります。': 'If you unset this, selects “Partial match” radio button by default.',
- },
- 'fr': {
- '完全一致': 'Concordance parfaite',
- '部分一致': 'Concordance partielle',
- 'タイトル・キャプション': 'Titre, Légende',
- '小説': 'Roman',
- 'タグ': 'Mots-clés',
- 'キーワード': 'Mots-clés',
- '本文': 'Contenu',
- 'ユーザー': 'Utilisateur',
- 'グループ': 'Groups',
- 'すべて': 'Tout',
- '検索': 'Rechercher',
- '小説検索': 'Rechercher un roman',
- 'ユーザー検索': '',
- 'グループ検索': 'Rechercher un groupe',
- 'pixiv 検索オプションを追加': '',
- 'この設定をオンにすると、ヘッダの高さが大きくなります。': '',
- '別のタブでpixivを開いていた場合、この設定の変更はそのタブの再読み込み後に反映されます。': '',
- 'イラスト・マンガの検索結果ページで、検索モードを維持': '',
- 'この設定をオフにすると、最初は「部分一致」にチェックが入った状態になります。': '',
- },
- 'ko': {
- '完全一致': '완전 일치',
- '部分一致': '부분 일치',
- 'タイトル・キャプション': '제목・캡션',
- '小説': '소설',
- 'タグ': '태그',
- 'キーワード': '키워드',
- '本文': '본문',
- 'ユーザー': '유저',
- 'グループ': '그룹',
- 'すべて': '전체',
- '検索': '검색',
- '小説検索': '소설 검색',
- 'ユーザー検索': '유저 검색',
- 'グループ検索': '그룹 검색',
- 'pixiv 検索オプションを追加': '',
- 'この設定をオンにすると、ヘッダの高さが大きくなります。': '',
- '別のタブでpixivを開いていた場合、この設定の変更はそのタブの再読み込み後に反映されます。': '',
- 'イラスト・マンガの検索結果ページで、検索モードを維持': '',
- 'この設定をオフにすると、最初は「部分一致」にチェックが入った状態になります。': '',
- },
- 'ru': {
- '完全一致': 'Полное совпадение',
- '部分一致': 'Частичное совпадение',
- 'タイトル・キャプション': 'Заголовок',
- '小説': 'Рассказы',
- 'タグ': 'Метка',
- 'キーワード': 'Ключевые слова',
- '本文': 'Текст',
- 'ユーザー': 'Пользователь',
- 'グループ': 'Группа',
- 'すべて': 'Все',
- '検索': 'Поиск',
- '小説検索': 'Искать рассказ',
- 'ユーザー検索': 'Искать пользователя',
- 'グループ検索': 'Искать группу',
- 'pixiv 検索オプションを追加': '',
- 'この設定をオンにすると、ヘッダの高さが大きくなります。': '',
- '別のタブでpixivを開いていた場合、この設定の変更はそのタブの再読み込み後に反映されます。': '',
- 'イラスト・マンガの検索結果ページで、検索モードを維持': '',
- 'この設定をオフにすると、最初は「部分一致」にチェックが入った状態になります。': '',
- },
- 'th': {
- '完全一致': '',
- '部分一致': '',
- 'タイトル・キャプション': 'ชื่อและคำบรรยาย',
- '小説': 'นิยาย',
- 'タグ': 'แท็ก',
- 'キーワード': 'คีย์เวิร์ด',
- '本文': '',
- 'ユーザー': 'ผู้ใช้',
- 'グループ': '',
- 'すべて': 'ทั้งหมด',
- '検索': 'ค้นหา',
- '小説検索': '',
- 'ユーザー検索': '',
- 'グループ検索': '',
- 'pixiv 検索オプションを追加': '',
- 'この設定をオンにすると、ヘッダの高さが大きくなります。': '',
- '別のタブでpixivを開いていた場合、この設定の変更はそのタブの再読み込み後に反映されます。': '',
- 'イラスト・マンガの検索結果ページで、検索モードを維持': '',
- 'この設定をオフにすると、最初は「部分一致」にチェックが入った状態になります。': '',
- },
- 'zh': {
- '完全一致': '完全相同',
- '部分一致': '部分相同',
- 'タイトル・キャプション': '题目/简述',
- '小説': '小说',
- 'タグ': '标签',
- 'キーワード': '关键词',
- '本文': '内容',
- 'ユーザー': '用户',
- 'グループ': '群组',
- 'すべて': '全部',
- '検索': '搜索',
- '小説検索': '搜索小说',
- 'ユーザー検索': '搜索用户',
- 'グループ検索': '搜索群组',
- 'pixiv 検索オプションを追加': '',
- 'この設定をオンにすると、ヘッダの高さが大きくなります。': '',
- '別のタブでpixivを開いていた場合、この設定の変更はそのタブの再読み込み後に反映されます。': '',
- 'イラスト・マンガの検索結果ページで、検索モードを維持': '',
- 'この設定をオフにすると、最初は「部分一致」にチェックが入った状態になります。': '',
- },
- 'zh-tw': {
- '完全一致': '完全相同',
- '部分一致': '部分相同',
- 'タイトル・キャプション': '題目/簡述',
- '小説': '小說',
- 'タグ': '標籤',
- 'キーワード': '關鍵詞',
- '本文': '內容',
- 'ユーザー': '用戶',
- 'グループ': '群組',
- 'すべて': '全部',
- '検索': '搜索',
- '小説検索': '搜索小說',
- 'ユーザー検索': '搜索用戶',
- 'グループ検索': '搜索群組',
- 'pixiv 検索オプションを追加': '',
- 'この設定をオンにすると、ヘッダの高さが大きくなります。': '',
- '別のタブでpixivを開いていた場合、この設定の変更はそのタブの再読み込み後に反映されます。': '',
- 'イラスト・マンガの検索結果ページで、検索モードを維持': '',
- 'この設定をオフにすると、最初は「部分一致」にチェックが入った状態になります。': '',
- },
- 'es': {
- '完全一致': 'Coincidencia exacta',
- '部分一致': 'Coincidencia parcial',
- 'タイトル・キャプション': 'Título/Descripción',
- '小説': 'Novelas',
- 'タグ': 'Etiquetas',
- 'キーワード': 'Palabra clave',
- '本文': 'Mensaje',
- 'ユーザー': 'Usuarios',
- 'グループ': 'Grupo',
- 'すべて': 'Todos',
- '検索': 'Buscar',
- '小説検索': 'Buscar novela',
- 'ユーザー検索': 'Buscar usuario',
- 'グループ検索': 'Buscar grupo',
- 'pixiv 検索オプションを追加': '',
- 'この設定をオンにすると、ヘッダの高さが大きくなります。': '',
- '別のタブでpixivを開いていた場合、この設定の変更はそのタブの再読み込み後に反映されます。': '',
- 'イラスト・マンガの検索結果ページで、検索モードを維持': '',
- 'この設定をオフにすると、最初は「部分一致」にチェックが入った状態になります。': '',
- },
- /*eslint-enable quote-props, max-len */
- });
-
- /**
- * スクリプトの起動。
- */
- class PixivSearchOptions
- {
- constructor()
- {
- this.id = Math.random();
-
- this.observeURLChanged();
-
- startScript(
- () => {
- if (document.body.classList.contains('not-logged-in')) {
- return;
- }
-
- /**
- * 検索窓。
- * @type {HTMLInputElement}
- * @access protected
- */
- this.word = document.querySelector(
- '#root form input[type="text"], #js-mount-point-header form input[type="text"]' // 後者は仕様変更前
- );
-
- this.main();
- },
- parent => parent.matches('#root, #js-mount-point-header div'), // 後者は仕様変更前
- target => target.matches('#root > div:first-child')
- || /* 仕様変更前 */ target.firstElementChild && target.firstElementChild.localName === 'form',
- () => document.querySelector(
- '#root form input[type="text"], #js-mount-point-header form input[type="text"]' // 後者は仕様変更前
- ),
- {},
- 60 * 1000
- );
- }
-
- observeURLChanged()
- {
- addEventListener('message', this);
-
- GreasemonkeyUtils.executeOnUnsafeContext(function (id) {
- History.prototype.pushState = new Proxy(History.prototype.pushState, {
- apply(target, thisArg, argumentsList)
- {
- postMessage({ id, url: argumentsList[2] }, location.origin);
- return Reflect.apply(target, thisArg, argumentsList);
- },
- });
- }, [this.id]);
- }
-
- /**
- * @param {Event}
- */
- handleEvent(event)
- {
- switch (event.type) {
- case 'message': {
- if (event.origin !== location.origin || typeof event.data !== 'object' || event.data === null
- || event.data.id !== this.id || !event.data.url) {
- break;
- }
- this.reflectSearchMode(new URL(event.data.url, location));
- break;
- }
- case 'submit': {
- const form = event.target;
- if (this.word.value === '') {
- break;
- }
-
- const searchMode = form.querySelector('[name="s_mode"]:checked');
- const subOption = form.querySelector('[name="s_sub_mode"]:checked');
-
- if (searchMode.value === 's_tag' || searchMode.value === 's_novel' && subOption.value === 's_tag') {
- // 検索窓のデフォルトの検索モードなら
- break;
- }
-
- event.stopPropagation();
-
- if (searchMode.dataset.formAction) {
- form.action = searchMode.dataset.formAction;
- }
- this.word.name = searchMode.dataset.wordName || 'word';
- if (searchMode.dataset.modeName) {
- searchMode.name = searchMode.dataset.modeName;
- }
-
- const novel = searchMode.value === 's_novel';
- if (subOption) {
- // 副検索モードが存在すれば
- searchMode.value = subOption.value;
- subOption.removeAttribute('name');
- }
-
- if (['s_usr', 's_group'].includes(searchMode.value)) {
- // ユーザー、グループの検索なら
- if (form.action !== location.origin + location.pathname) {
- // サービス間をまたぐ場合、不要なパラメーターは送信しない
- for (const hiddenParam of form.querySelectorAll('[type="hidden"]')) {
- hiddenParam.disabled = true;
- }
- }
- break;
- }
-
- const result = /^\/tags\/[^/]+\/([a-z]+)$/.exec(location.pathname);
- const currentMode = result ? result[1] : '';
- if (!currentMode || currentMode !== 'novels' && novel || currentMode === 'novels' && !novel
- || decodeURIComponent(/^\/tags\/([^/]+)/u.exec(location.pathname)[1]) !== this.word.value) {
- // 検索結果以外のページ、サービス間をまたぐ場合、
- // またはキーワードが現在のページと異なる場合
- form.action = `/tags/${encodeURIComponent(this.word.value)}/${novel ? 'novels' : 'artworks'}`;
- this.word.removeAttribute('name');
- if (searchMode.value === 's_tag_full') {
- // タグ完全一致なら、s_modeを削除
- searchMode.removeAttribute('name');
- }
- break;
- }
-
- // イラストからイラスト、小説から小説の場合
- event.preventDefault();
-
- // 検索オプションボタンをクリック
- document.querySelector('[d^="M0 1C0 0.447754"]').closest('button').click();
- new MutationObserver((mutations, observer) => {
- let dialog;
- for (const mutation of mutations) {
- dialog = Array.from(mutation.addedNodes).find(
- node => node.nodeType === Node.ELEMENT_NODE && node.getAttribute('role') === 'presentation'
- );
- if (dialog) {
- break;
- }
- }
- if (!dialog) {
- return;
- }
- observer.disconnect();
-
- dialog.hidden = true;
-
- // 対象
- const input = dialog.querySelectorAll('[class$="-dummyInput"]')[novel ? 0 : 1];
- // プルダウンメニューを開く
- input.dispatchEvent(new KeyboardEvent('keydown', { bubbles: true, key: ' ' }));
- new MutationObserver(function (mutations, observer) {
- let select, options;
- mutations: for (const mutation of mutations) {
- for (const node of mutation.addedNodes) {
- options = node.querySelectorAll('[role="option"]');
- if (options.length === 0) {
- continue;
- }
- select = node;
- break mutations;
- }
- }
- if (!select) {
- return;
- }
- observer.disconnect();
-
- // 項目を選択する
- options[(novel
- ? ['s_tag_only', 's_tag_full', 's_tc', 's_tag']
- : ['s_tag', 's_tag_full', 's_tc']).indexOf(searchMode.value)].click();
- new MutationObserver(function (mutations, observer) {
- if (!mutations.some(mutation => Array.from(mutation.removedNodes).includes(select))) {
- return;
- }
- observer.disconnect();
-
- setTimeout(function () {
- // 「適用する」ボタンを押す
- dialog.querySelector('[type="submit"]').click();
- }, 100);
- }).observe(select.parentElement, { childList: true });
- }).observe(dialog, { subtree: true, childList: true });
- }).observe(document.body, { childList: true });
- break;
- }
- case 'change': {
- // 副検索モードの選択
- const subOptions = event.target.parentElement.parentElement;
- if (event.target.name === 's_mode') {
- // 検索モードの選択なら
- if (subOptions.classList.contains('sub-options')) {
- // 副検索モードが存在すれば
- subOptions.querySelector('[value="s_tag"], [value="keyword_all"]').checked = true;
- } else {
- const subOption = event.currentTarget.querySelector('[name="s_sub_mode"]:checked');
- if (subOption) {
- subOption.checked = false;
- }
- }
- this.word.placeholder = event.target.dataset.placeholder;
- } else if (event.target.name === 's_sub_mode') {
- // 副検索モードの選択なら
- const subOption = subOptions.firstElementChild.firstElementChild;
- subOption.checked = true;
- this.word.placeholder = subOption.dataset.placeholder;
- }
- break;
- }
- case 'dblclick':
- // ラベルをダブルクリックで検索
- if (event.target.matches('label, label *')) {
- event.currentTarget.dispatchEvent(new Event('submit'));
- }
- break;
- }
- }
-
- main()
- {
- // 言語の設定
- Gettext.setLocale(document.documentElement.lang);
-
- /**
- * 検索オプションを設定するラジオボタンのHTML文字列。
- * @type {string}
- */
- const optionsHTML = h`
- <label>
- <input data-placeholder="${_('検索')}" value="s_tag_full" name="s_mode" type="radio" />
- ${_('完全一致')}
- </label>
- <label>
- <input data-placeholder="${_('検索')}" value="s_tag" name="s_mode" type="radio" />
- ${_('部分一致')}
- </label>
- <label>
- <input data-placeholder="${_('検索')}" value="s_tc" name="s_mode" type="radio" />
- ${_('タイトル・キャプション')}
- </label>
- <div id="s_novel" class="sub-options" title="${_('小説')}">
- <label>
- <input data-placeholder="${_('小説検索')}" value="s_novel" name="s_mode" type="radio" />
- <img alt="${_('小説')}" src="" />
- </label>
- <label>
- <input value="s_tag_full" name="s_sub_mode" type="radio" />
- ${_('タグ')}
- </label>
- <label>
- <input value="s_tag" name="s_sub_mode" type="radio" />
- ${_('キーワード')}
- </label>
- <label>
- <input value="s_tc" name="s_sub_mode" type="radio" />
- ${_('本文')}
- </label>
- </div>
- <label title="${_('ユーザー')}">
- <input data-placeholder="${_('ユーザー検索')}" data-word-name="nick"
- data-form-action="https://www.pixiv.net/search_user.php" value="s_usr" name="s_mode" type="radio" />
- <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 2048 2048" width="16px" height="16px">
- <title>${_('ユーザー')}</title>
- <path fill="#809DB8" d="M848,348 l98,-20 q28,0 54,22 q38,32 48,44 q22,26 34,26 q16,0 26,-4 q14,-6 52,2 q32,6 48,24 q18,20 23,42 t5,50 q0,32 2,48 q-4,24 -2,38 q6,36 6,62 t-6,44 t12,28 t38,10 h22 q28,0 40,46 q8,30 8,60 q0,14 -2,30 l-6,46 q-2,16 -6,22 q-6,10 -10,16 t-14,14 l-18,12 l-24,16 q-18,10 -30,18 q-22,62 -50,102 q-34,48 -58,70 l-22,22 q-10,96 -10,118 v6 q0,8 2,16 q0,8 4,22 q4,18 12,28 l20,28 q10,14 30,28 q8,6 46,26 q26,14 62,24 q46,12 82,18 q88,12 138,28 q60,18 102,44 t62,40 t50,46 q28,30 30,32 q20,22 40,98 q20,74 20,140 v66 h-1768 v-72 q0,-42 14,-126 q16,-92 34,-106 q8,-6 32,-34 t46,-48 t62,-46 q36,-24 100,-44 q66,-22 140,-32 q60,-8 102,-22 q40,-12 74,-32 t50,-34 q26,-24 34,-36 q10,-16 16,-30 q12,-18 10,-20 q-4,-6 -2,-6 v-132 q-4,-2 -12,-8 q-8,-4 -28,-22 q-14,-10 -40,-36 q-18,-18 -38,-50 t-32,-64 l-36,-18 q-20,-10 -24,-14 q-4,-2 -12,-18 q-8,-14 -8,-16 q0,-4 -2,-18 t-4,-22 q0,-4 2,-10 t5,-26 t3,-38 q0,-20 8,-38 q10,-32 12,-36 q8,-14 18,-24 t22,-10 h14 q8,-2 18,-12 q12,-12 8,-30 q2,-64 2,-134 q0,-22 8,-50 q10,-38 18,-58 q12,-32 38,-56 t60,-32 l52,-12 h-2 q2,0 6,-2 q10,-2 12,-2 v-6 v-4 q-2,-4 0,-6 q0,-2 8,-4 t22,-2 z" />
- </svg>
- </label>
- <div id="s_group" class="sub-options" title="${_('グループ')}">
- <label>
- <input data-placeholder="${_('グループ検索')}" data-mode-name="mode"
- data-form-action="https://www.pixiv.net/group/search_group.php"
- value="s_group" name="s_mode" type="radio" />
- <img alt="${_('グループ')}" src="" />
- </label>
- <label>
- <input value="keyword_all" name="s_sub_mode" type="radio" />
- ${_('すべて')}
- </label>
- <label>
- <input value="tag_full" name="s_sub_mode" type="radio" />
- ${_('完全一致')}
- </label>
- <label>
- <input value="tag" name="s_sub_mode" type="radio" />
- ${_('部分一致')}
- </label>
- <label>
- <input value="keyword" name="s_sub_mode" type="radio" />
- ${_('タイトル・キャプション')}
- </label>
- </div>`;
-
- const form = this.word.form;
- form.id = 'suggest-container';
-
- this.style({
- gridElementSelector: Array.from(form.parentElement.parentElement.classList).map(cls => '.' + cls).join(''),
- });
-
- // 検索オプションを設定するラジオボタンの追加
- form.insertAdjacentHTML('beforeend', optionsHTML);
-
- form.parentElement.addEventListener('submit', this, true);
- form.addEventListener('change', this);
- form.addEventListener('dblclick', this);
-
- // 現在の検索モードを設定
- this.reflectSearchMode(location);
- }
-
- /**
- * 現在のページの検索モードをもとに、検索窓の検索モードを設定します。
- * @param {string} url
- * @returns {void}
- */
- reflectSearchMode(url)
- {
- const { pathname, searchParams } = new URL(url);
-
- let mode, subMode;
-
- const form = this.word.form;
-
- if (pathname.startsWith('/tags/')) {
- mode = searchParams.get('s_mode');
- if (pathname.endsWith('novels')) {
- // 小説の検索結果
- subMode = mode;
- mode = 's_novel';
- } else {
- // イラストの検索結果
- const currentMode = mode || 's_tag_full';
- mode = 's_tag';
-
- GM.getValue('check-match-full-by-default-on-full-match-result', false).then(function (value) {
- if (value) {
- form.querySelector(`[value="${currentMode}"]`).click();
- }
- });
- }
- } else if (pathname.startsWith('/novel/')) {
- mode = 's_novel';
- } else if (pathname === '/search_user.php') {
- mode = 's_usr';
- } else if (pathname.startsWith('/group/')) {
- mode = 's_group';
- } else {
- mode = 's_tag';
- }
-
- form.querySelector(!subMode ? `[value="${mode}"]` : `#${mode} [value="${subMode}"]`).click();
- }
-
- /**
- * スタイルシートを設定します。
- * @access protected
- */
- style({ gridElementSelector })
- {
- document.head.insertAdjacentHTML('beforeend', `<style>
- ${gridElementSelector} {
- grid-template-columns: 1fr minmax(auto, 970px) 1fr;
- }
-
- #suggest-container {
- display: flex;
- align-items: center;
- justify-content: center;
- }
-
- /*------------------------------------
- 検索オプション
- */
- #suggest-container > div:first-of-type {
- padding-right: 1em;
- }
- #suggest-container label {
- padding: 0 0.7em;
- display: flex;
- align-items: center;
- white-space: nowrap;
- }
- #suggest-container label input {
- margin-right: 0.3em;
- width: initial;
- }
-
- /*------------------------------------
- 副検索モード
- */
- #suggest-container .sub-options label:not(:first-of-type) {
- display: none;
- position: absolute;
- z-index: 1;
- width: 13em;
- height: 2em;
- border: solid 1px #D6DEE5;
- border-top-style: none;
- border-bottom-style: none;
- background: #FFFFFF;
- }
- #suggest-container .sub-options label:nth-of-type(2) {
- border-top-style: solid;
- border-radius: 5px 5px 0 0;
- }
- #suggest-container .sub-options label:nth-of-type(3) {
- margin-top: 2em;
- }
- #suggest-container .sub-options label:nth-of-type(4) {
- margin-top: 4em;
- }
- #suggest-container .sub-options label:nth-of-type(5) {
- margin-top: 6em;
- }
- #suggest-container .sub-options label:last-of-type {
- border-bottom-style: solid;
- border-radius: 0 0 5px 5px;
- }
- #suggest-container .sub-options:hover label {
- display: flex;
- }
- #suggest-container .sub-options:hover::after {
- display: block;
- }
- </style>`);
- }
- }
-
- const pixivSearchOptions = new PixivSearchOptions();
-
- /**
- * 設定を管理します。
- */
- class PixivSearchOptionsSettings
- {
- constructor()
- {
- startScript(
- async () => {
- GM.registerMenuCommand(_('pixiv 検索オプションを追加'), () => this.showDialog());
- },
- parent => parent === document.documentElement,
- target => target === document.body,
- () => document.body
- );
- }
-
- /**
- * 設定ダイアログを表示します。
- */
- async showDialog()
- {
- if (!this.dialog) {
- this.initialize();
- }
-
- const form = document.forms['pixiv-search-options-settings'];
- form['check-match-full-by-default-on-full-match-result'].checked
- = await GM.getValue('check-match-full-by-default-on-full-match-result', false);
-
- this.dialog.showModal();
- }
-
- /**
- * @param {Event} event
- */
- handleEvent(event)
- {
- switch (event.type) {
- case 'change':
- GM.setValue(event.target.name, event.target.checked);
- break;
-
- case 'click':
- this.dialog.close();
- break;
- }
- }
-
- /**
- * 設定ダイアログを構築します。
- * @access private
- */
- initialize()
- {
- document.head.insertAdjacentHTML('beforeend', `<style>
- /*====================================
- pixiv 検索オプションを表示 設定ダイアログ
- */
- #pixiv-search-options-settings {
- border-radius: 0.5em;
- border-color: #69AFCA;
- border-width: 0.5em;
- }
-
- #pixiv-search-options-settings h1 {
- font-size: 1.5em;
- font-weight: bold;
- margin-bottom: 0.5em;
- text-align: center;
- }
-
- #pixiv-search-options-settings form {
- color: #333333;
- }
-
- #pixiv-search-options-settings form ul li {
- margin: 0.2em 0;
- }
-
- #pixiv-search-options-settings form label {
- display: flex;
- align-items: center;
- }
-
- #pixiv-search-options-settings form small {
- margin-left: 1.3em;
- }
-
- #pixiv-search-options-settings form [name="close"] {
- border: none;
- position: absolute;
- border-radius: 50%;
- position: absolute;
- top: -1em;
- right: -1em;
- width: 2.5em;
- height: 2.5em;
- background:
- black url("https://source.pixiv.net/www/images/common/icon_modal_close.png") no-repeat 50% 50%;
- cursor: pointer;
- overflow: hidden;
- white-space: nowrap;
- text-indent: 200%;
- }
- </style>`);
-
- document.body.insertAdjacentHTML('afterbegin', h`<dialog id="pixiv-search-options-settings">
- <h1>${_('pixiv 検索オプションを追加')}</h1>
- <form name="pixiv-search-options-settings">
- <ul>
- <li>
- <label>
- <input type="checkbox" name="check-match-full-by-default-on-full-match-result" />
- ${_('イラスト・マンガの検索結果ページで、検索モードを維持')}
- </label>
- <p><small class="settingColor">${_('この設定をオフにすると、最初は「部分一致」にチェックが入った状態になります。')}</small></p>
- </li>
- </ul>
- <button type="button" name="close">${_('閉じる')}</button>
- </form>
- </dialog>`);
-
- this.dialog = document.getElementById('pixiv-search-options-settings');
- this.polyfillDialogElements([this.dialog]);
-
- const form = document.forms['pixiv-search-options-settings'];
- form.addEventListener('change', this);
- form.close.addEventListener('click', this);
- }
-
- /**
- * Microsoft Edge、およびFirefox向けに、dialog要素のpolyfillを行います。
- * @access private
- * @param {HTMLDialogElement[]} dialogs
- * @returns {Promise.<void>}
- */
- async polyfillDialogElements(dialogs)
- {
- document.head.insertAdjacentHTML(
- 'beforeend',
- h`<link rel="stylesheet" href="${await GM.getResourceUrl('dialog-polyfill.css')}" />`
- );
-
- for (const dialog of dialogs) {
- /*globals dialogPolyfill */
- dialogPolyfill.registerDialog(dialog);
- }
- }
- }
-
- new PixivSearchOptionsSettings();