您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
[userChromeJS] Drag selected character strings or image and drop to the semitransparent box displayed on web page to open search result.
当前为
// ==UserScript== // @name Drag & DropZones + // @name:ja Drag & DropZones + // @description [userChromeJS] Drag selected character strings or image and drop to the semitransparent box displayed on web page to open search result. // @description:ja 【userChromeJS】選択した文字列などをドラッグし、ページ上に表示される半透明の枠内にドロップすることで、Web検索などを実行する // @namespace https://userscripts.org/users/347021 // @version 3.0.3 // @include main // @license Mozilla Public License Version 2.0 (MPL 2.0); https://www.mozilla.org/MPL/2.0/ // @compatible Firefox userChromeJS用スクリプト です (※Greasemonkeyスクリプトではありません) // @incompatible Opera // @incompatible Chrome // @author 100の人 // @contributor HADAA // @homepage https://greasyfork.org/scripts/264 // ==/UserScript== (function () { 'use strict'; /** * L10N * @type {LocalizedTexts} */ let localizedTexts = { 'en': { 'Drag & DropZones +': 'Drag & DropZones +', '検索エンジン名': 'Search engine name', '他と重複しないエンジン名を入力してください。': 'Please input search engine names without repetition.', 'URL・POSTパラメータ': 'URL, POST parameters', 'POSTパラメータの設定': 'Setting POST parameters', '名前': 'name', '値': 'value', 'メソッド': 'Method', 'データの種類': 'Data type', '文字列': 'String', '画像': 'Image', '音声': 'Audio', '文字符号化方式': 'Character encoding scheme', 'キャンセル': 'Cancel', 'OK': 'OK', '行を追加': 'Add row', '行を削除': 'Delete row', '上に新しい行を挿入': 'Insert new row above', '下に新しい行を挿入': 'Insert new row below', '上に新しい行を挿入します。': 'Insert new row above.', '下に新しい行を挿入します。': 'Insert new row below.', '上の行に移動します。': 'Move focus to above row.', '下の行に移動します。': 'Move focus to below row.', '行をドラッグ & ドロップで、順番を変更できます。': 'Drag and drop row to change order.', 'アイコンボタンのポップアップメニューから、アイコンを変更できます。検索窓のエンジンのアイコンは変更できません。': 'You can change icon via the icon button pop-up menu. You cannot modify the search engine icon in the search bar.', 'アイコンを変更': 'Modify icon', '元のアイコンに戻す': 'Restore to default icon', 'ローカルファイルからアイコンを設定': 'Set icon from local file', '画像ファイルを選択してください。': 'Please choose image file.', 'Webページ、または画像ファイルのURLからアイコンを設定': 'Set icon by URL of Web page or image file', 'Webページ、または画像ファイルのURLを入力してください。': 'Please input URL of Web page or image file.', 'アイコンの設定に失敗しました。約 %s KiB までの画像を設定できます。': 'Setting icon failed. Image up to about %s KiB can be set.', 'クリップボードのURL、または画像データからアイコンを設定': 'Set icon by URL or image data on clipboard', 'クリップボードからデータを取得できませんでした。': 'Could not get data from clipboard.', '指定されたURLに接続できませんでした。': 'Connection to specified URL failed.', 'http:// などで始まるURLを入力してください。': 'Specify URL starting with "http://" etc.', 'アイコンを取得できませんでした。WebページのURLであれば、一度ブラウザでページを表示してみてください。': 'Could not get icon. If you have input a URL of a Web page, please open that page in your browser once.', 'アイコンを一括取得': 'Collectively get icons', 'アイコン未取得の検索エンジンについて、URLを基にアイコンを取得します。アイコンボタンのポップアップメニューの「元のアイコンに戻す」から、個別に取得することもできます。': 'Get icons from URL for search engines without an icon. You can choose "Restore to default icon" from the icon button pop-up menu to get individual ones.', 'アイコンの取得が完了しました。': 'Completed getting icons.', '検索窓のエンジンの追加': 'Add engine in Search Bar', '選択してください': 'Choose', '検索結果を開く場所': 'Where to open search result', '現在のタブ。Ctrl、Shiftキーを押していれば、それぞれ新しいタブ、ウィンドウ': 'Current tab. If Ctrl or Shift key is pressed, it will open in a new tab or window, respectively', '新しいタブ': 'New tab', '新しいウィンドウ': 'New window', '検索窓に新しい検索エンジンが追加されたとき、自動的にドロップゾーンとしても追加する。': 'When new engine is added to Search Bar, a dropzone will also be automatically created.', 'テキスト入力欄のキーボードショートカット': 'Keyboard Shortcuts in input box', 'または': 'or', 'インポートとエクスポート': 'Import and export', 'インポート': 'Import', '現在の設定をすべて削除し、XMLファイルから設定をインポートします。ブラウザの検索エンジンサービスに同名の検索エンジンが存在する場合は、そちらを優先します。': 'Delete all settings, then import settings from XML file. If the browser search service with the same name already exists, the existing one takes priority.', '%s からのインポートに失敗しました。': 'Import from "%s" failed.', // %sはファイル名 'XMLパースエラーです。': 'XML parse error occured.', '検索エンジンが一つも見つかりませんでした。': 'Not even one search engine was found.', '%s からのインポートが完了しました。': 'Import from "%s" completed.', // %sはファイル名 'エクスポート': 'Export', '現在の設定をファイルへエクスポートします。保存していない設定は反映されません。': 'Export current settings to file. Not yet saved settings are not reflected.', '%s へ設定をエクスポートしました。': 'Export to "%s" completed.', // %sはファイルパス '追加インポート': 'Additional import', 'XMLファイルから検索エンジンを追加します。同名の検索エンジンがすでに存在する場合は上書きします。': 'Add search engine from XML file. If a search engine with the same name already exists, overwrite it.', 'インポートした設定を保存するには、「OK」ボタンをクリックしてください。': 'Click "OK" button to save import data.', 'JSON文字列から追加インポート': 'Additional import from JSON string', '本スクリプトのバージョン1でエクスポートしたJSON文字列から、検索エンジンを追加します。': 'Add search engine from JSON string exported by version 1 of this script.', 'JSON文字列を貼り付けてください。': 'Please paste JSON string.', 'JSON文字列からのインポートに失敗しました。': 'Import from JSON string failed.', 'JSONパースエラーです。': 'JSON parse error occured.', 'JSON文字列からのインポートが完了しました。': 'Import form JSON string completed.', 'その他': 'Others', '設定を初期化': 'Initialize the settings', 'すべての設定を削除し、初回起動時の状態に戻します。': 'Delete all settings, then restore to first starting state.', '本当に、『%s』のすべての設定を削除してもよろしいですか?': 'Are you sure you want to delete all settings of "%s" ?', // %sは当スクリプト名 '設定の初期化が完了しました。': 'Settings initialization completed.', 'すべての設定を削除': 'Delete all settings', 'すべての設定を削除し、スクリプトを停止します。': 'Delete all settings, and stop this script.', '設定の削除が完了しました。当スクリプト自体を削除しなければ、次回のブラウザ起動時にまた設定が作成されます。': 'Completed deleting all settings. If you don\'t delete this script, settings will be created again when you start your browser next time.', 'Google 画像で検索': 'Google search by image', }, }; Cu.import('resource://gre/modules/FileUtils.jsm'); let ScriptableUnicodeConverter = Cc['@mozilla.org/intl/scriptableunicodeconverter'].createInstance(Ci.nsIScriptableUnicodeConverter); let NativeJSON = Cc['@mozilla.org/dom/json;1'].createInstance(Ci.nsIJSON); let DOMSerializer = Cc['@mozilla.org/xmlextras/xmlserializer;1'].createInstance(Ci.nsIDOMSerializer); let TextToSubURI = Cc['@mozilla.org/intl/texttosuburi;1'].getService(Ci.nsITextToSubURI); let MIMEService = Cc['@mozilla.org/mime;1'].getService(Ci.nsIMIMEService); let FaviconService = Cc['@mozilla.org/browser/favicon-service;1'].getService(Ci.nsIFaviconService); let ImgTools = Cc['@mozilla.org/image/tools;1'].getService(Ci.imgITools); let StringInputStream = Components.Constructor('@mozilla.org/io/string-input-stream;1', 'nsIStringInputStream', 'setData'); let MultiplexInputStream = Components.Constructor('@mozilla.org/io/multiplex-input-stream;1', 'nsIMultiplexInputStream'); let MIMEInputStream = Components.Constructor('@mozilla.org/network/mime-input-stream;1', 'nsIMIMEInputStream'); let FileInputStream = Components.Constructor('@mozilla.org/network/file-input-stream;1', 'nsIFileInputStream', 'init'); let BinaryInputStream = Components.Constructor('@mozilla.org/binaryinputstream;1', 'nsIBinaryInputStream', 'setInputStream'); let FilePicker = Components.Constructor('@mozilla.org/filepicker;1', 'nsIFilePicker', 'init'); let Transferable = Components.Constructor('@mozilla.org/widget/transferable;1', 'nsITransferable', 'addDataFlavor'); let Cr = new Proxy(window.Cr, { get: function (target, name, receiver) { if (name in target) { return target[name]; } else if (name === 'NS_ERROR_UCONV_NOCONV') { return 0x80500001; } else { return undefined; } }, }); // i18n let _, gettext, setlang, setLocalizedTexts; { /** * 翻訳対象文字列 (msgid) の言語。 * @constant {string} */ let ORIGINAL_LOCALE = 'ja'; /** * クライアントの言語の翻訳リソースが存在しないとき、どの言語に翻訳するか。 * @constant {string} */ let DEFAULT_LOCALE = 'en'; /** * 以下のような形式の翻訳リソース。 * { * 'IETF言語タグ': { * '翻訳前 (msgid)': '翻訳後 (msgstr)', * …… * }, * …… * } * @typedef {Object} LocalizedTexts */ /** * クライアントの言語。{@link setlang}から変更される。 * @type {string} * @access private */ let langtag = 'ja'; /** * クライアントの言語のlanguage部分。{@link setlang}から変更される。 * @type {string} * @access private */ let language = 'ja'; /** * 翻訳リソース。{@link setLocalizedTexts}から変更される。 * @type {LocalizedTexts} * @access private */ let multilingualLocalizedTexts = {}; multilingualLocalizedTexts[ORIGINAL_LOCALE] = {}; /** * テキストをクライアントの言語に変換する。 * @param {string} message - 翻訳前。 * @returns {string} 翻訳後。 */ _ = gettext = function (message) { // クライアントの言語の翻訳リソースが存在すれば、それを返す return langtag in multilingualLocalizedTexts && multilingualLocalizedTexts[langtag][message] // 地域下位タグを取り除いた言語タグの翻訳リソースが存在すれば、それを返す || language in multilingualLocalizedTexts && multilingualLocalizedTexts[language][message] // デフォルト言語の翻訳リソースが存在すれば、それを返す || DEFAULT_LOCALE in multilingualLocalizedTexts && multilingualLocalizedTexts[DEFAULT_LOCALE][message] // そのまま返す || message; }; /** * {@link gettext}から参照されるクライアントの言語を設定する。 * @param {string} lang - IETF言語タグ。(「language」と「language-REGION」にのみ対応) */ setlang = function (lang) { lang = lang.split('-', 2); language = lang[0].toLowerCase(); langtag = language + (lang[1] ? '-' + lang[1].toUpperCase() : ''); }; /** * {@link gettext}から参照される翻訳リソースを追加する。 * @param {LocalizedTexts} localizedTexts */ setLocalizedTexts = function (localizedTexts) { var localizedText, lang, language, langtag, msgid; for (lang in localizedTexts) { localizedText = localizedTexts[lang]; lang = lang.split('-'); language = lang[0].toLowerCase(); langtag = language + (lang[1] ? '-' + lang[1].toUpperCase() : ''); if (langtag in multilingualLocalizedTexts) { // すでに該当言語の翻訳リソースが存在すれば、統合する(同じmsgidがあれば上書き) for (msgid in localizedText) { multilingualLocalizedTexts[langtag][msgid] = localizedText[msgid]; } } else { multilingualLocalizedTexts[langtag] = localizedText; } if (language !== langtag) { // 言語タグに地域下位タグが含まれていれば // 地域下位タグを取り除いた言語タグも翻訳リソースとして追加する if (language in multilingualLocalizedTexts) { // すでに該当言語の翻訳リソースが存在すれば、統合する(同じmsgidがあれば無視) for (msgid in localizedText) { if (!(msgid in multilingualLocalizedTexts[language])) { multilingualLocalizedTexts[language][msgid] = localizedText[msgid]; } } } else { multilingualLocalizedTexts[language] = localizedText; } } // msgidの言語の翻訳リソースを生成 for (msgid in localizedText) { multilingualLocalizedTexts[ORIGINAL_LOCALE][msgid] = msgid; } } }; } setLocalizedTexts(localizedTexts); setlang(window.navigator.language); /** * スクリプトの中核。 */ let DragAndDropZonesPlus = { /** * id属性値などに利用する識別子。 * @constant {string} */ ID: 'drag-and-drop-search-347021', /** * スクリプト名。 * @constant {string} */ NAME: _('Drag & DropZones +'), /** * ウィンドウが開いたときに実行する処理。 */ main: function () { // ドロップゾーンの作成 let earliestWindow = Services.wm.getEnumerator('navigator:browser').getNext(); if (earliestWindow === window) { // ブラウザ起動時 // 検索エンジンサービスの初期化を待機 Services.search.init(function () { if (SearchUtils.getDropzoneLength() < 1) { // 検索エンジンが1つも登録されていなければ(初回起動なら)、検索窓のエンジンを登録する SearchUtils.initializeDropzones(); } DropzoneUtils.create(); DropzoneUtils.setEventListeners(); }); SettingsScreen.addToMenu(); BrowserSearchEngineModifiedObserver.init(); } else { // 新しいウィンドウを開いたとき let originalWrapper = earliestWindow.document.getElementById(DragAndDropZonesPlus.ID); if (!originalWrapper) { // 最初に開かれたウィンドウでスクリプトが実行されていなければ、終了 return; } DropzoneUtils.wrapper = originalWrapper.cloneNode(true); let appContent = document.getElementById('appcontent'); appContent.insertBefore(DropzoneUtils.wrapper, appContent.firstChild); DropzoneUtils.setEventListeners(); DropzoneUtils.resetDropzones(true); SettingsScreen.addToMenu(); } ObserverUtils.init(this.ID); UninstallObserver.init(); }, /** * 設定を初期化する。 */ initialize: function () { gPrefService.getBranch(SettingsUtils.ROOT_BRANCH_NAME).deleteBranch(''); SearchUtils.initializeDropzones(); DropzoneUtils.update(); }, /** * 設定を削除し、スクリプトを停止する。 */ uninstall: function () { gPrefService.getBranch(SettingsUtils.ROOT_BRANCH_NAME).deleteBranch(''); UninstallObserver.notify(); }, }; /** * 一つの検索エンジンを表す。 * @typedef {Object} SearchEngine * @property {number} [index] - prefs.jsに保存されている場合のインデックス。 * @property {string} [icon] - 検索エンジンを表す16px×16pxのアイコンのDataURL。 * @property {string} name - 検索エンジン名。 * @property {boolean} browserSearchEngine - ブラウザの検索エンジンの情報ならtrue。 * @property {string} url - 検索エンジンに結果をリクエストするときのURL。 * @property {string} [method] - 検索エンジンが受け入れるHTTPメソッド。GETメソッドかPOSTメソッド。 * @property {FormDataEntry[]} [params] - 検索エンジンがPOSTメソッドを受け入れる場合のPOSTパラメータ。 * @property {string} [accept] - 検索エンジンが受け入れるデータの種類。text/*、image/*、audio/*のいずれか。 * @property {string} [encoding] - 検索エンジンが受け入れる文字コード文字符号化方式。 */ /** * 検索エンジンに関する操作群。 */ let SearchUtils = { /** * XMLパースエラーを示す要素の名前空間。 * @constant {string} */ PARSE_ERROR_NS: 'http://www.mozilla.org/newlayout/xml/parsererror.xml', /** * 保存可能なドロップゾーンの最大数。 * @constant {number} */ MAX_DROPZONE_LENGTH: 32, /** * prefs.jsからドロップゾーンの検索エンジン数を取得する。(歯抜けインデックスを含む) * @returns {number} */ getDropzoneLength: function () { let branch = gPrefService.getBranch(SettingsUtils.ROOT_BRANCH_NAME + 'engines.'); let indexes = []; for (let prefName of branch.getChildList('')) { let [index, property] = prefName.split('.'); if (property !== undefined && /^(?:0|[1-9][0-9]*)$/.test(index) && index < this.MAX_DROPZONE_LENGTH) { indexes.push(index); } else { // 壊れた設定なら、削除する branch.clearUserPref(prefName); } } return indexes.length > 0 ? Math.max.apply(null, indexes) + 1 : 0; }, /** * prefs.jsに検索窓のエンジンを登録する。 */ initializeDropzones: function () { let engines = Services.search.getVisibleEngines().map(function (engine) { return { name: engine.name, }; }); // POST検索例 engines.push({ icon: '', name: _('Google 画像で検索'), url: 'https://www.google.com/searchbyimage/upload', method: 'POST', params: [['encoded_image', '{searchTerms}']], accept: 'image/*', encoding: StringUtils.THE_ENCODING, }); SearchUtils.setEngines(engines); }, /** * 検索エンジン名からブラウザに登録されているエンジンを取得する。 * @param {string} name * @returns {?SearchEngine} - encodingプロパティを含まない。 */ getBrowserEngineByName: function (name) { let browserEngine = Services.search.getEngineByName(name); return browserEngine ? this.convertEngineFromBrowser(browserEngine) : null; }, /** * {@link Ci.nsISearchEngine}を{@link SearchEngine}に変換する。 * @param {Ci.nsISearchEngine} browserEngine * @returns {SearchEngine} - encodingプロパティを含まない。 */ convertEngineFromBrowser: function (browserEngine) { let engine = { browserSearchEngine: true, name: browserEngine.name, accept: 'text/plain', }; if (browserEngine.iconURI) { engine.icon = browserEngine.iconURI.spec; } let searchTerms = String(Math.random()).replace('.', ''); let submission = browserEngine.getSubmission(searchTerms); if (submission.postData) { // POSTメソッドなら engine.method = 'POST'; engine.url = submission.uri.spec; let postData = NetUtil.readInputStreamToString(submission.postData, submission.postData.available()); engine.params = Array.from( new URLSearchParams(postData.split('\r\n\r\n')[1].replace(searchTerms, '{searchTerms}')) ); } else { // GETメソッドなら engine.method = 'GET'; engine.url = submission.uri.spec.replace(new RegExp(searchTerms, 'g'), '{searchTerms}'); } return engine; }, /** * インデックスからprefs.jsに保存されているユーザー定義のエンジンを取得する。 * @param {number} index * @returns {?SearchEngine} */ getCustomEngineByIndex: function (index) { let engine = null; let branch = new Preferences(SettingsUtils.ROOT_BRANCH_NAME + 'engines.' + index + '.'); let name = branch.get('name'); if (name) { engine = { index: index, browserSearchEngine: false, name: name, url: branch.get('url', ''), method: branch.get('method', 'GET'), accept: branch.get('accept', 'text/plain'), encoding: branch.get('encoding', StringUtils.THE_ENCODING), }; let icon = branch.get('icon'); if (icon) { engine.icon = icon; } if (engine.method === 'POST') { // POSTメソッドなら let params; try { params = JSON.parse(branch.get('params', '[]')); } catch (e) { if (!(e instanceof SyntaxError)) { throw e; } } engine.params = []; if (Array.isArray(params)) { for (let param of params) { if (Array.isArray(param)) { engine.params.push([param[0] || '', param[1] || '']); } } } } } return engine; }, /** * インデックスからエンジンを取得する。 * @param {number} index * @returns {?SearchEngine} */ getEngineByIndex: function (index) { let branch = new Preferences(SettingsUtils.ROOT_BRANCH_NAME + 'engines.' + index + '.'); return branch.get('url') ? this.getCustomEngineByIndex(index) : this.getBrowserEngineByName(branch.get('name')); }, /** * prefs.jsに保存されている検索エンジンをすべて取得する。 * @returns {SearchEngine[]} */ getEngines: function () { let encodings = this.getBrowserEngineEncodings(); let engines = []; for (let i = 0, l = this.getDropzoneLength(); i < l; i++) { let branch = new Preferences(SettingsUtils.ROOT_BRANCH_NAME + 'engines.' + i + '.'); if (branch.get('url')) { // ユーザー定義のエンジンなら engines.push(this.getCustomEngineByIndex(i)); } else { // ブラウザのエンジンなら let name = branch.get('name'); let engine = this.getBrowserEngineByName(name); if (engine) { engine.index = i; engine.encoding = encodings[name] || StringUtils.THE_ENCODING; engines.push(engine); } } } return engines; }, /** * ブラウザに登録されている検索エンジンをすべて取得する。 * @returns {SearchEngine[]} */ getBrowserEngines: function () { let encodings = this.getBrowserEngineEncodings(); return Services.search.getVisibleEngines().map(browserEngine => { let engine = this.convertEngineFromBrowser(browserEngine); engine.encoding = encodings[engine.name] || StringUtils.THE_ENCODING; return engine; }); }, /** * prefs.jsに保存されているユーザー定義のエンジンをすべて取得する。 * @returns {SearchEngine[]} */ getCustomEngines: function () { let engines = []; for (let i = 0, l = this.getDropzoneLength(); i < l; i++) { let branch = new Preferences(SettingsUtils.ROOT_BRANCH_NAME + 'engines.' + i + '.'); if (branch.get('url')) { // ユーザー定義のエンジンなら engines.push(this.getCustomEngineByIndex(i)); } } return engines; }, /** * prefs.jsに保存されている検索エンジンをすべて削除し、指定したエンジンリストと置き換える。 * @param {SearchEngine[]} engines */ setEngines: function (engines) { let oldEngineLength = this.getDropzoneLength(); engines.forEach((engine, index) => { this.setEngine(index, engine); }); let branch = gPrefService.getBranch(SettingsUtils.ROOT_BRANCH_NAME + 'engines.'); for (let i = engines.length; i < oldEngineLength; i++) { branch.deleteBranch(i + '.'); } }, /** * prefs.jsのブランチの指定した位置に検索エンジンを追加する。 * @param {number} index * @param {SearchEngine} engine */ setEngine: function (index, engine) { // 古い設定の削除 gPrefService.getBranch(SettingsUtils.ROOT_BRANCH_NAME + 'engines.').deleteBranch(index + '.'); let branch = new Preferences(SettingsUtils.ROOT_BRANCH_NAME + 'engines.' + index + '.'); branch.set('name', engine.name); if (!Services.search.getEngineByName(engine.name)) { // 同名の検索エンジンがブラウザに存在しなければ if (engine.icon) { branch.set('icon', engine.icon); } branch.set('url', engine.url); if (engine.method !== 'GET') { // POSTメソッドなら branch.set('method', engine.method); branch.set('params', JSON.stringify(engine.params)); if (engine.accept !== 'text/plain') { branch.set('accept', engine.accept); } } if (engine.encoding !== StringUtils.THE_ENCODING) { branch.set('encoding', engine.encoding); } } }, /** * 長すぎる値を切り詰める。 * @param {SearchEngine} engine * @returns {SearchEngine} */ trimValues: function (engine) { for (let name in engine) { switch (name) { case 'icon': delete engine[name]; break; case 'params': while (JSON.stringify(engine[name]) > SettingsUtils.MAX_PREFERENCE_VALUE_LENGTH) { engine[name].pop(); } break; default: if (typeof engine[name] === 'string' && engine[name] > SettingsUtils.MAX_PREFERENCE_VALUE_LENGTH) { engine[name] = engine[name].substr(0, SettingsUtils.MAX_PREFERENCE_VALUE_LENGTH); } } } return engine; }, /** * ファイル選択ダイアログを表示し、選択されたXMLファイルの検索エンジンを返す。 * ファイルが選択されなかった場合は何もしない。 * @param {Window} win - ダイアログの親となるウィンドウ。 * @param {Function} callback - 第1引数に{@link SearchEngine[]}、第2引数にファイルの文書。第3引数にファイル名。エラーが起きていれば、第4引数にエラーメッセージ。 */ getSearchEnginesFromFile: function (win, callback) { let filePicker = new FilePicker(win, null, Ci.nsIFilePicker.modeOpen); filePicker.appendFilters(Ci.nsIFilePicker.filterXML); filePicker.appendFilters(Ci.nsIFilePicker.filterAll); filePicker.open(result => { if (result === Ci.nsIFilePicker.returnOK) { let client = new XMLHttpRequest(); client.open('GET', NetUtil.newURI(filePicker.file).spec); client.responseType = 'document'; client.addEventListener('load', event => { let doc = event.target.response; let root = doc.documentElement; if (root.namespaceURI === this.PARSE_ERROR_NS && root.localName === 'parseerror') { // パースエラーが起きていれば callback(null, null, filePicker.file.leafName, _('XMLパースエラーです。') + '\n\n' + root.textContent); } else { let engines = []; for (let description of doc.getElementsByTagNameNS(OpenSearchUtils.NS, 'OpenSearchDescription')) { let engine = OpenSearchUtils.convertEngineToObject(description); if (engine) { engines.push(engine); } } if (engines.length > 0) { callback(engines, doc, filePicker.file.leafName); } else { // 検索エンジンが一つも含まれていなければ callback(null, null, filePicker.file.leafName, _('検索エンジンが一つも見つかりませんでした。')); } } }); client.send(); } }); }, /** * search.jsonから、ブラウザの検索エンジンの文字符号化方式を取得する。 * @returns {Object} プロパティ名に検索エンジン名、値に文字符号化方式をもつオブジェクト。 * @access protected */ getBrowserEngineEncodings: function () { let encodings = {}; let file = FileUtils.getFile('ProfD', ['search.json']); if (file.exists()) { let stream = new FileInputStream(file, -1, -1, 0); try { let searchJson = NativeJSON.decodeFromStream(stream, -1); if (searchJson && searchJson.directories) { for (let directory in searchJson.directories) { if (directory && Array.isArray(directory.engines)) { for (let engine of directory.engines) { encodings[engine._name] = engine.queryCharset; } } } } } catch (e) { if (!(e instanceof SyntaxError)) { throw e; } } finally { stream.close(); } } return encodings; }, }; /** * prefs.jsなどの設定を読み書きする。 */ let SettingsUtils = { /** * エクスポートするXMLファイルで使用する名前空間。 * @constant {string} */ NS: 'https://userscripts.org/scripts/show/130510', /** * 設定を保存するブランチ名。(末尾にピリオドを含む) * @constant {string} */ ROOT_BRANCH_NAME: 'extensions.' + DragAndDropZonesPlus.ID + '.', /** * prefs.jsの一項目の容量制限。(UTF-16のコードユニット数) * @constant {number} */ MAX_PREFERENCE_VALUE_LENGTH: 1 * 1024 * 1024, /** * ファイルに設定をエクスポートする。 * @param {Window} win - ダイアログの親となるウィンドウ。 * @param {Function} [callback] - 第1引数にファイルのフルパス。 */ exportToFile: function (win, callback = function () { }) { // ファイル保存ダイアログを開く let filePicker = new FilePicker(win, null, Ci.nsIFilePicker.modeSave); filePicker.appendFilters(Ci.nsIFilePicker.filterXML); filePicker.appendFilters(Ci.nsIFilePicker.filterAll); filePicker.defaultString = DragAndDropZonesPlus.ID + '.xml'; filePicker.open(result => { if (result === Ci.nsIFilePicker.returnOK || result === Ci.nsIFilePicker.returnReplace) { let settingsDocument = new Document(); let settings = settingsDocument.createElementNS(this.NS, 'settings'); let engines = settingsDocument.createElementNS(this.NS, 'engines'); for (let engine of SearchUtils.getEngines()) { engines.appendChild(OpenSearchUtils.convertEngineToElement(engine, settingsDocument)); } settings.appendChild(engines); let branch = new Preferences(this.ROOT_BRANCH_NAME); let where = branch.get('where', 'tab'); settings.appendChild(settingsDocument.createElementNS(this.NS, 'where')).textContent = where; let automaticallyReflect = branch.get('automaticallyReflect', true); if (automaticallyReflect) { settings.appendChild(settingsDocument.createElementNS(this.NS, 'automatically-reflect')).textContent = 'on'; } settingsDocument.appendChild(settings); // 保存 let stream = FileUtils.openSafeFileOutputStream(filePicker.file); DOMSerializer.serializeToStream(DOMUtils.toPrettyXML(settings), stream, ''); FileUtils.closeSafeFileOutputStream(stream); callback(filePicker.file.path); } }); }, /** * ファイルから設定をインポートする。 * @param {Window} win - ダイアログの親となるウィンドウ。 * @param {Function} [callback] - 第1引数にファイル名。インポートに失敗していれば、第2引数にエラーメッセージ。 */ importFromFile: function (win, callback = function () { }) { SearchUtils.getSearchEnginesFromFile(win, (engines, settingsDocument, fileName, errorMessage) => { if (engines) { SearchUtils.setEngines(engines); let branch = new Preferences(this.ROOT_BRANCH_NAME); let where = settingsDocument.getElementsByTagNameNS(this.NS, 'where')[0]; if (where) { let value = where.textContent.trim(); if (value !== 'tab') { branch.set('where', value); } } let automaticallyReflect = settingsDocument.getElementsByTagNameNS(this.NS, 'automatically-reflect')[0]; if (!automaticallyReflect || automaticallyReflect.textContent.trim() !== 'on') { branch.set('automaticallyReflect', false); } DropzoneUtils.update(); callback(fileName); } else { callback(fileName, errorMessage); } }); }, }; /** * バージョン1の設定値。 */ let Version1Settings = { /** * JSON文字列から検索エンジンを取得する。 * @param {Window} win - ダイアログの親となるウィンドウ。 * @param {Function} [callback] - 取得に成功していれば、第1引数に検索エンジンの配列。失敗していれば、第2引数にエラーメッセージ。 */ getEnginesFromText: function (win, callback) { let jsonString = win.prompt(_('JSON文字列を貼り付けてください。')); if (jsonString && jsonString.trim()) { let errorMessage = _('JSON文字列からのインポートに失敗しました。'); let oldSettings; try { oldSettings = JSON.parse(jsonString); } catch (e) { if (e instanceof SyntaxError) { callback(null, _('JSONパースエラーです。')); return; } else { throw e; } } if (Array.isArray(oldSettings)) { let doc = win.document; // ブラウザのエンジン名一覧を取得しておく let browserEngineNames = Array.prototype.map.call(doc.querySelectorAll('[name=add-browser-engine] > [label]'), function (option) { return option.label; }); // 利用可能な文字コード一覧を取得しておく let encodings = Array.prototype.map.call(doc.getElementsByTagName('template')[0].content.querySelectorAll('[name=encoding] > option'), function (option) { return option.value; }); // デフォルトのFaviconのDataURLを取得しておく let client = new XMLHttpRequest(); client.open('GET', DropzoneUtils.DEFAULT_ICON); client.responseType = 'blob'; client.addEventListener('load', function (event) { IconUtils.convertToDataURL(event.target.response, function (defaultIconDataURL) { let engines = []; for (let oldSetting of oldSettings) { if (typeof oldSetting === 'object' && oldSetting !== null && oldSetting.title) { let engine = { name: oldSetting.title, }; if (browserEngineNames.indexOf(engine.name) !== -1) { // 同名の検索エンジンがブラウザに存在すれば engines.push(engine) } else if (oldSetting.query) { engine.url = oldSetting.query + '{searchTerms}'; if (oldSetting.icon && oldSetting.icon !== defaultIconDataURL) { engine.icon = oldSetting.icon; } if (oldSetting.encoding && encodings.indexOf(oldSetting.encoding) !== -1) { engine.encoding = oldSetting.encoding; } engines.push(engine); } } } if (engines.length > 0) { callback(engines); } else { callback(null, _('検索エンジンが一つも見つかりませんでした。')); } }); }); client.send(); } } else { callback(null, _('検索エンジンが一つも見つかりませんでした。')); } }, }; /** * OpenSearchに関する操作群。 */ let OpenSearchUtils = { /** * OpenSearchの名前空間。 * @constant {string} */ NS: 'http://a9.com/-/spec/opensearch/1.1/', /** * OpenSearch Referrer extensionの名前空間。 * @constant {string} */ REFERRER_NS: 'http://a9.com/-/opensearch/extensions/referrer/1.0/', /** * OpenSearch parameter extensionの名前空間。 * @constant {string} */ PARAMETER_NS: 'http://a9.com/-/spec/opensearch/extensions/parameters/1.0/', /** * OpenSearch Suggestions extensionの名前空間。 * @constant {string} */ SUGGESTIONS_NS: 'http://www.opensearch.org/specifications/opensearch/extensions/suggestions/1.1', /** * OpenSearch Geo extensionの名前空間。 * @constant {string} */ GEO_NS: 'http://a9.com/-/opensearch/extensions/geo/1.0/', /** * OpenSearch Time Extensionの名前空間。 * @constant {string} */ TIME_NS: 'http://a9.com/-/opensearch/extensions/time/1.0/', /** * OpenSearch Mobile Extensionの名前空間。 * @constant {string} */ M_NS: 'http://a9.com/-/opensearch/extensions/mobile/1.0/', /** * OpenSearch SRU Extensionの名前空間。 * @constant {string} */ SRU_NS: 'http://a9.com/-/opensearch/extensions/sru/2.0/', /** * OpenSearch Semantic Extensionの名前空間。 * @constant {string} */ SEMANTIC_NS: 'http://a9.com/-/opensearch/extensions/semantic/1.0/', /** * InputEncoding要素の既定値。 * @constant {string} */ DEFAULT_ENCODING: 'UTF-8', /** * itemsPerPage要素が存在しない場合の、countパラメータの既定値。 * @constant {number} */ DEFAULT_ITEMS_PER_PAGE: 20, /** * {@link SearchEngine}をOpenSearchDescription要素に変換する。 * @param {SearchEngine} engine * @param {XMLDocument} doc - 作成するOpenSearchDescription要素のノード文書。 * @returns {Element} OpenSearchDescription要素。 */ convertEngineToElement: function (engine, doc) { let description = doc.createElementNS(this.NS, 'OpenSearchDescription'); description.appendChild(doc.createElementNS(this.NS, 'ShortName')).textContent = engine.name; description.appendChild(doc.createElementNS(this.NS, 'Description')); if (engine.icon) { description.appendChild(doc.createElementNS(this.NS, 'Image')).textContent = engine.icon; } let url = doc.createElementNS(this.NS, 'Url'); url.setAttribute('template', engine.url); url.setAttribute('type', 'text/html'); if (engine.method === 'POST') { // POSTメソッドなら url.setAttributeNS(this.PARAMETER_NS, 'parameters:method', engine.method); for (let [name, value] of engine.params) { let parameter = doc.createElementNS(this.PARAMETER_NS, 'parameters:Parameter'); parameter.setAttribute('name', name); parameter.setAttribute('value', value); url.appendChild(parameter); } if (engine.accept !== 'text/plain') { url.setAttributeNS(SettingsUtils.NS, 'dnd-search:accept', engine.accept); } } description.appendChild(url); if (engine.encoding !== this.DEFAULT_ENCODING) { description.appendChild(doc.createElementNS(OPEN_SEARCH_NS, 'InputEncoding')).textContent = encoding; } return description; }, /** * OpenSearchDescription要素を{@link SearchEngine}に変換する。 * @param {Element} description - OpenSearchDescription要素。 * @returns {?SearchEngine} */ convertEngineToObject: function (description) { let engine = null; let shortName = description.getElementsByTagNameNS(this.NS, 'ShortName')[0]; let url = description.querySelector('Url[template][type="text/html"], Url[template][type="application/xhtml+xml"]') || description.querySelector('Url[template]'); if (shortName && url) { let template = this.parseURLTemplate(url.getAttribute('template'), url); let nativeURL; try { nativeURL = NetUtil.newURI(template).QueryInterface(Ci.nsIURL); } catch (e) { if (e.result === Cr.NS_ERROR_MALFORMED_URI || e.result === Cr.NS_NOINTERFACE) { // 妥当なURLでなければ return null; } else { throw e; } } engine = { name: shortName.textContent, }; let image = description.getElementsByTagNameNS(this.NS, 'Image')[0]; if (image) { engine.icon = image.textContent; } let parameters = url.getElementsByTagNameNS(this.PARAMETER_NS, 'Parameter'); let method = url.getAttributeNS(this.PARAMETER_NS, 'method'); if (method && method.toUpperCase() === 'POST') { // POSTメソッドなら engine.method = 'POST'; engine.url = template; engine.params = []; for (let parameter of parameters) { engine.params.push([ parameter.getAttribute('name'), this.parseURLTemplate(parameter.getAttribute('value'), parameter)]); } engine.accept = url.getAttributeNS(SettingsUtils.NS, 'accept') || 'text/*'; } else { // POSTメソッド以外はGETメソッドとして扱う engine.method = 'GET'; let searchParams = new URLSearchParams(nativeURL.query); for (let parameter of parameters) { searchParams.append( parameter.getAttribute('name'), this.parseURLTemplate(parameter.getAttribute('value'), parameter)); } nativeURL.query = searchParams.toString().replace(/%7BsearchTerms%7D/g, '{searchTerms}'); engine.url = nativeURL.spec; engine.accept = 'text/*'; } let inputEncoding = description.getElementsByTagNameNS(this.NS, 'InputEncoding')[0]; engine.encoding = inputEncoding ? inputEncoding.textContent : this.DEFAULT_ENCODING; if (engine.encoding !== this.DEFAULT_ENCODING && engine.encoding.toUpperCase() === this.DEFAULT_ENCODING) { engine.encoding = this.DEFAULT_ENCODING; } } return engine || SearchUtils.trimValues(engine); }, /** * OpenSearch URLテンプレートに含まれるテンプレートパラメータのうち、{searchTerms}以外を置換する。 * @param {string} template - OpenSearch URLテンプレート。 * @param {Element} element - OpenSearch URLテンプレートが設定されているUrl要素、またはParameter要素。 * @returns {string} * @see [OpenSearch URL template syntax]{@link http://www.opensearch.org/Specifications/OpenSearch/1.1#OpenSearch_URL_template_syntax} * @access protected */ parseURLTemplate: function (template, element) { let description = DOMUtils.getParentElementByTagName(element, 'OpenSearchDescription'); let url = DOMUtils.getParentElementByTagName(element, 'Url'); let searchValue = /{((?:[-a-zA-Z0-9._~!$&'()*+,;=:@]|%[0-9A-Fa-f]{2})+:)?((?:[-a-zA-Z0-9._~!$&'()*+,;=:@]|%[0-9A-Fa-f]{2})*)(\?)?}/g; return template.replace(searchValue, (parameter, encodedPrefix, encodedLname, modifier) => { let prefix = decodeURIComponent(encodedPrefix), lname = decodeURIComponent(encodedLname); let value = null; switch (prefix ? element.lookupNamespaceURI(prefix) || this.NS : this.NS) { case this.NS: switch (lname) { case 'searchTerms': return '{searchTerms}'; case 'count': if (modifier) { value = ''; } else { let itemsPerPage = description.getElementsByTagNameNS(this.NS, 'itemsPerPage'); value = itemsPerPage ? itemsPerPage.textContent : this.DEFAULT_ITEMS_PER_PAGE; } break; case 'startIndex': value = url.getAttribute('indexOffset') || 1; break; case 'startPage': value = url.getAttribute('pageOffset') || 1; break; case 'language': value = window.navigator.language; break; case 'inputEncoding': let encoding = description.getElementsByTagNameNS(this.NS, 'InputEncoding'); value = encoding ? encoding.textContent : this.DEFAULT_ENCODING; break; case 'outputEncoding': value = StringUtils.THE_ENCODING; break; } break; case this.REFERRER_NS: switch (lname) { case 'source': value = DragAndDropZonesPlus.ID; break; } break; case this.SUGGESTIONS_NS: switch (lname) { //case 'suggestionPrefix': // break; case 'suggestionIndex': value = modifier ? '' : 0; break; } break; //case this.GEO_NS: // break; case this.TIME_NS: switch (lname) { case 'start': value = modifier ? '' : '0000-01-01T00:00:00Z'; break; case 'end': value = modifier ? '' : '9999-12-31T23:59:58Z'; break; } break; case this.M_NS: switch (lname) { case 'userAgent': value = window.navigator.userAgent; break; //case 'subId': // break; //case 'mcc': // break; //case 'mnc': // break; } break; case this.SRU_NS: switch (lname) { case 'queryType': value = modifier ? '' : 'searchTerms'; break; case 'query': return modifier ? '' : '{searchTerms}'; case 'startRecord': value = modifier ? '' : url.getAttribute('indexOffset') || 1; break; case 'maximumRecords': if (modifier) { value = ''; } else { let itemsPerPage = description.getElementsByTagNameNS(this.NS, 'itemsPerPage'); value = itemsPerPage ? itemsPerPage.textContent : this.DEFAULT_ITEMS_PER_PAGE; } break; case 'recordPacking': value = modifier ? '' : 'xml'; break; //case 'recordSchema': // break; //case 'resultSetTTL': // break; //case 'sortKeys': // break; //case 'stylesheet': // break; case 'rendering': value = 'server'; break; case 'httpAccept': value = 'text/html, application/xhtml+xml'; break; case 'httpAcceptCharset': value = modifier ? '' : '*'; break; case 'httpAcceptEncoding': value = modifier ? '' : '*'; break; case 'httpAcceptLanguage': value = modifier ? '' : window.navigator.language + ', *'; break; case 'httpAcceptRanges': value = modifier ? '' : 'none'; break; case 'facetLimit': value = modifier ? '' : 1; break; case 'facetSort': value = modifier ? '' : 'recordCount'; break; //case 'facetRangeField': // break; //case 'facetLowValue': // break; //case 'facetHighValue': // break; //case 'facetCount': // break; //case 'extension': // break; } break; //case this.SEMANTIC_NS: // break; } return value === null ? (modifier ? '' : parameter) : encodeURIComponent(value); }); }, }; /** * アイコンに関する操作群。 */ let IconUtils = { /** * SQLiteのLIKE演算子におけるESCAPE文字。 * @constant {string} */ SQLITE_LIKE_ESCAPE_STRING: '@', /** * DataURLに変換する。 * @param {(Blob|Array)} icon * @param {Function} callback - 第1引数にDataURL。 * @param {string} [type] - iconが配列の場合の、アイコンのMIMEタイプ。 */ convertToDataURL: function (icon, callback, type) { if (type) { icon = new Blob([new Uint8Array(icon)], { type: type }); } let reader = new FileReader(); reader.addEventListener('load', function (event) { callback(event.target.result); }); reader.readAsDataURL(icon); }, /** * URLから、そのWebサイトのFaviconのDataURLを取得する。 * @param {string} url * @param {Function} callback - 第1引数にDataURL。 */ getFaviconFromSiteUrl: function (url, callback) { let nativeURL = NetUtil.newURI(url).QueryInterface(Ci.nsIURL); PlacesUtils.favicons.getFaviconDataForPage(nativeURL, (faviconURL, length, data, type) => { if (length > 0) { // Faviconが存在すれば this.convertToDataURL(data, callback, type); } else { // places.sqliteに接続する let places = Services.storage.openDatabase(FileUtils.getFile('ProfD', ['places.sqlite'])); // 指定されたURLに似た履歴のFaviconを取得するSQL文を構築・実行 let statement = places.createAsyncStatement('SELECT data, mime_type' + ' FROM moz_places INNER JOIN moz_favicons ON favicon_id = moz_favicons.id' + ' WHERE moz_places.url LIKE :url ESCAPE :escape ORDER BY last_visit_date DESC LIMIT 1'); statement.params.url = statement.escapeStringForLIKE(nativeURL.prePath + nativeURL.directory, this.SQLITE_LIKE_ESCAPE_STRING) + '%'; statement.params.escape = this.SQLITE_LIKE_ESCAPE_STRING; let favicon; statement.executeAsync({ handleResult: resultSet => { let favicon = resultSet.getNextRow(); statement.finalize(); }, handleError: error => { this.getFaviconIco(nativeURL.prePath, callback); }, handleCompletion: () => { if (favicon) { this.convertToDataURL(favicon.getResultByName('data'), callback, favicon.getResultByName('mime_type')); } else { this.getFaviconIco(nativeURL.prePath, callback); } }, }); places.asyncClose(); } }); }, /** * Faviconを含むWebページのURL、または画像のURLからBlobインスタンスを取得する。 * @param {string} url * @param {Function} callback - 第1引数にDataURL。エラーが起こった場合は、第2引数にエラーメッセージ。 */ getFromUrl: function (url, callback) { let uri; try { uri = NetUtil.newURI(url); } catch (e) { if (e.result === Cr.NS_ERROR_MALFORMED_URI) { // 妥当なURLでなければ callback(null, _('http:// などで始まるURLを入力してください。')); return; } else { throw e; } } let client = new XMLHttpRequest(); try { client.open('GET', url); } catch (e) { if (e.result === Cr.NS_ERROR_UNKNOWN_PROTOCOL) { callback(null, _('http:// などで始まるURLを入力してください。')); return; } else { throw e; } } client.responseType = 'blob'; client.addEventListener('error', () => { this.getFaviconFromSiteUrl(url, callback, function () { callback(null, _('指定されたURLに接続できませんでした。')); }); }); client.addEventListener('load', event => { let client = event.target; if (client.status === 200) { if (client.response.type.startsWith('image/')) { this.convertToDataURL(client.response, callback); } else { this.getFaviconFromSiteUrl(url, callback, function () { callback(null, _('アイコンを取得できませんでした。WebページのURLであれば、一度ブラウザでページを表示してみてください。')); }); } } else { this.getFaviconFromSiteUrl(url, callback, function () { callback(null, _('指定されたURLに接続できませんでした。') + '\n' + client.status + ' ' + client.statusText); }); } }); client.send(); }, /** * ローカルファイルのDataURLを取得する。 * @param {Window} win - ダイアログの親となるウィンドウ。 * @param {Function} callback - 第1引数にDataURL。エラーが起こった場合は、第2引数にエラーメッセージ。 */ getFromLocalFile: function (win, callback) { // ファイル選択ダイアログを開く let filePicker = new FilePicker(win, null, Ci.nsIFilePicker.modeOpen); filePicker.appendFilters(Ci.nsIFilePicker.filterImages); filePicker.open(result => { if (result === Ci.nsIFilePicker.returnOK) { let file = filePicker.file; let type; try { // 拡張子を元にMIMEタイプを取得 type = MIMEService.getTypeFromFile(file); } catch(e) { if (e.result === Cr.NS_ERROR_NOT_AVAILABLE) { // 未知の拡張子 callback(null, _('画像ファイルを選択してください。')); } else { throw e; } } if (type.startsWith('image/')) { // Blobインスタンスを取得 let fileStream = new FileInputStream(file, -1, -1, 0); let stream = new BinaryInputStream(fileStream); this.convertToDataURL(stream.readByteArray(stream.available()), callback, type); stream.close(); fileStream.close(); } else { callback(null, _('画像ファイルを指定してください。')); } } }); }, /** * クリップボードのURL、または画像からDataURLを取得する。 * @param {Function} callback - 第1引数にDataURL。エラーが起こった場合は、第2引数にエラーメッセージ。 */ getIconFromClipboard: function (callback) { if (Services.clipboard.supportsSelectionClipboard()) { // 選択クリップボードが有効なOSなら let url = ClipboardUtils.getText(Services.clipboard.kSelectionClipboard); if (url) { // テキストデータが保持されていれば this.getFromUrl(url, (dataURL, errorMessage) => { if (dataURL) { callback(dataURL); } else { this.getIconFromGlobalClipboard(callback); } }); } else { this.getIconFromGlobalClipboard(callback); } } else { this.getIconFromGlobalClipboard(callback); } }, /** * クリップボード(グローバル)のURL、または画像からDataURLを取得する。 * @param {Function} callback - 第1引数にDataURL。エラーが起こった場合は、第2引数にエラーメッセージ。 * @access protected */ getIconFromGlobalClipboard: function (callback) { let url = ClipboardUtils.getText(Services.clipboard.kGlobalClipboard); if (url) { // テキストデータが保持されていれば this.getFromUrl(url, callback); return; } else if (Services.clipboard.hasDataMatchingFlavors(['image/png'], 1, Services.clipboard.kGlobalClipboard)) { // 画像データが保持されていれば、PNG画像として取得する(Windowsでは透過部分が黒色になる) // <http://mxr.mozilla.org/mozilla-central/source/addon-sdk/source/lib/sdk/clipboard.js#259>を参考 let transferable = new Transferable('image/png'), data = {}; Services.clipboard.getData(transferable, Services.clipboard.kGlobalClipboard); transferable.getTransferData('image/png', data, {}); let image = data.value; if (image instanceof Ci.nsISupportsInterfacePointer) { image = image.data; } if (image instanceof Ci.imgIContainer) { image = ImgTools.encodeImage(image, 'image/png'); } if (image instanceof Ci.nsIInputStream) { this.convertToDataURL(new BinaryInputStream(image).readByteArray(image.available()), callback, 'image/png'); return; } } callback(null, _('クリップボードからデータを取得できませんでした。')); }, /** * /favicon.ico を取得する。 * @param {string} origin * @param {Function} callback - 第1引数にDataURL。 * @access protected */ getFaviconIco: function (origin, callback) { let client = new XMLHttpRequest(); client.open('GET', origin + '/' + 'favicon.ico'); client.responseType = 'blob'; client.addEventListener('error', () => callback(null)); client.addEventListener('load', event => { let client = event.target; if (client.status === 200 && client.response.type.startsWith('image/')) { this.convertToDataURL(client.response, callback); } else { callback(null); } }); client.send(); }, }; /** * ドロップゾーンの作成やドロップされたデータの検索などを行う。 * @type {Object} */ let DropzoneUtils = { /** * 設定されていない場合に表示するアイコンのURL。 * @constant {string} * @see Ci.nsIFaviconService#defaultFavicon */ DEFAULT_ICON: 'resource://gre/chrome/toolkit/skin/classic/mozapps/places/defaultFavicon.png', /** * ドロップゾーン専用のスタイルシートを設定するための親要素。 * @type {HTMLDivElement} */ wrapper: null, /** * 各ドロップゾーンを作成。 */ create: function () { this.wrapper = document.createElementNS(DOMUtils.HTML_NS, 'div'); this.wrapper.id = DragAndDropZonesPlus.ID; this.wrapper.hidden = true; let style = document.createElementNS(DOMUtils.HTML_NS, 'style'); style.scoped = true; this.wrapper.appendChild(style); this.wrapper.appendChild(document.createElementNS(DOMUtils.HTML_NS, 'ul')); let appContent = document.getElementById('appcontent'); appContent.insertBefore(this.wrapper, appContent.firstChild); this.update(); }, /** * ドロップゾーンを初期状態に戻す。 * @param {boolean} [forced] - {@link DropzoneUtils.itemTypesDuringDrag}の確認を行わずに実行するなら真。 */ resetDropzones: function (forced = false) { if (forced || this.itemTypesDuringDrag) { let activeValidDropzone = this.getActiveValidDropzone(); if (activeValidDropzone) { activeValidDropzone.classList.remove('drop-active-valid'); } this.wrapper.hidden = true; this.itemTypesDuringDrag = null; this.dragstartEvent = null; this.dragoverEventAlreadyFired = true; } }, /** * ドロップゾーンに関するスタイルシートとイベントリスナーを設定する。 */ setEventListeners: function () { let styleSheet = this.wrapper.getElementsByTagNameNS(DOMUtils.HTML_NS, 'style')[0].sheet; let cssRules = styleSheet.cssRules; [ // 位置決め用 'div {' + 'position: relative;' + '}', // ドロップゾーン全体 'ul {' + 'position: absolute;' + 'top: 1.5em;' + 'left: 1.5em;' + 'right: 1.5em;' + 'height: 8em;' + 'display: flex;' + 'border: solid #A0A0A0 1px;' + 'background-color: rgba(100, 200, 255, 0.5);' + 'padding-left: 0;' + '}', // 各ドロップゾーン 'li {' + 'flex: 1;' + 'font-weight: bold;' + 'padding-left: 0.5em;' + 'overflow: hidden;' + 'white-space: nowrap;' + 'line-height: 2em;' + 'position: relative;' + 'z-index: 1;' + '}', 'li:not(:first-of-type) {' + 'border-left: inherit;' + '}', 'img {' + 'width: 16px;' + 'height: 16px;' + 'vertical-align: middle;' + 'margin-right: 0.3em;' + '}', // ドロップゾーン上部の背景色 'li::before {' + 'display: block;' + 'content: "";' + 'position: absolute;' + 'top: 0;' + 'left: 0;' + 'right: 0;' + 'height: 2em;' + 'background-color: rgba(50, 100, 200, 0.7);' + 'z-index: -1;' + '}', // 各ドロップゾーンにポインタが載っている時 'li.drop-active-valid::before {' + 'height: initial;' + 'bottom: 0;' + '}', ].forEach(function (rule) { styleSheet.insertRule(rule, cssRules.length); }); // dropzone属性の代替 // Bug 723008 – Implement dropzone content attribute <https://bugzilla.mozilla.org/show_bug.cgi?id=723008> this.wrapper.addEventListener('dragover', event => { let activeValidDropzone = this.getActiveValidDropzone(); if (activeValidDropzone && activeValidDropzone.contains(event.target)) { event.preventDefault(); } }); // イベントリスナーの追加 for (let type of this.eventTypesForWindow) { window.addEventListener(type, this, true); } }, /** * windowに追加したイベントリスナーを取り除く。 */ removeEventListeners: function () { for (let type of this.eventTypesForWindow) { window.removeEventListener(type, this, true); } }, /** * :drop(active valid)な要素にdrop-active-validクラスを追加する。 * @param {HTMLElement} target - :drop(active valid)か否か調べる要素。 */ setActiveValidDropzone: function (target) { if (target.nodeType === Node.ELEMENT_NODE && this.wrapper.contains(target)) { // ドロップゾーンなら let dropzone = DOMUtils.getAttributeAsDOMTokenList(target, 'dropzone'); if (dropzone.contains('link') && Array.prototype.some.call(dropzone, type => this.itemTypesDuringDrag.indexOf(type) !== -1)) { // 各ドロップゾーンにポインタが載った時、 // ドロップゾーンが受け取ることができるデータをドラッグしていれば target.classList.add('drop-active-valid'); } } }, /** * イベントハンドラ。 * @param {Event} event */ handleEvent: function (event) { let target = event.target; switch (event.type) { case 'dragstart': if (event.isTrusted) { // ユーザーによるドラッグなら if (this.itemTypesDuringDrag) { // ドロップゾーンが表示されたままなら this.resetDropzones(); } this.itemTypesDuringDrag = this.getItemTypes(event); if (this.itemTypesDuringDrag.length > 0) { // ドロップゾーンを表示 this.wrapper.hidden = false; this.dragstartEvent = event; this.dragoverEventAlreadyFired = false; } } break; case 'dragover': if (!this.dragoverEventAlreadyFired) { this.dragoverEventAlreadyFired = true; // ドラッグ開始時、すでにドロップゾーン内にカーソルがあった場合、dragenterイベントが発生しないため if (target.nodeType === Node.ELEMENT_NODE) { this.setActiveValidDropzone(target); } } break; case 'dragenter': if (event.relatedTarget) { let activeValidDropzone = this.getActiveValidDropzone(); if (activeValidDropzone && !activeValidDropzone.contains(target)) { // 各ドロップゾーンからポインタが外れた時 activeValidDropzone.classList.remove('drop-active-valid'); } this.setActiveValidDropzone(target); } else { // ウィンドウ外からのドラッグなら if (this.itemTypesDuringDrag) { if (this.itemTypesDuringDrag.length > 0) { this.wrapper.hidden = false; } } else { // ドラッグ開始なら if (event.isTrusted) { this.itemTypesDuringDrag = ['string:text/plain', 'file:text/*', 'file:image/*', 'file:audio/*']; // ドロップゾーンを表示 this.wrapper.hidden = false; } } } break; case 'dragleave': if (this.itemTypesDuringDrag && !event.relatedTarget && !this.wrapper.hidden) { // ウィンドウ外へドラッグされたとき this.wrapper.hidden = true; let activeValidDropzone = this.getActiveValidDropzone(); if (activeValidDropzone) { activeValidDropzone.classList.remove('drop-active-valid'); } } break; case 'dragend': this.resetDropzones(); break; case 'drop': if (this.wrapper.contains(target)) { // 各ドロップゾーンにドロップされた時 event.preventDefault(); let dropzone = DOMUtils.getAttributeAsDOMTokenList(target, 'dropzone'); if (this.dragstartEvent) { if (dropzone.contains('file:image/*')) { // 画像としてドロップしたとき this.getMIMEInputStreamFromURL(this.getImageURLFromDragstartEvent(this.dragstartEvent), mimeInputStream => this.searchDropData(mimeInputStream, target.dataset.engineIndex, event)); } else { // 文字列としてドロップしたとき this.searchDropData(this.getTextFromDragstartEvent(this.dragstartEvent), target.dataset.engineIndex, event); } } else { // ウィンドウ外からのドロップ if (dropzone.contains('file:image/*') || dropzone.contains('file:audio/*')) { // 画像、または音声ファイルとしてドロップしたとき let type = dropzone.contains('file:image/*') ? 'image' : 'audio'; let fileType = event.dataTransfer.files[0].type; if (fileType.startsWith(type + '/')) { // ドロップゾーンが受け取ることができる形式のファイルなら let file = event.dataTransfer.mozGetDataAt('application/x-moz-file', 0); if (file instanceof Ci.nsIFile) { let mimeInputStream = new MIMEInputStream(); mimeInputStream.addHeader('content-type', fileType); mimeInputStream.setData(new FileInputStream(file, -1, -1, 0)); this.searchDropData(mimeInputStream, target.dataset.engineIndex, event); } } } else { // 文字列としてドロップしたとき let text = this.getTextFromDropEvent(event, !dropzone.contains('file:text/*')); if (text) { this.searchDropData(text, target.dataset.engineIndex, event); } } } } this.resetDropzones(); break; } }, /** * prefs.jsの設定値を元に、各ウィンドウのドロップゾーンを更新する。 */ update: function () { // 構築 let dropzones = new DocumentFragment(); for (let engine of SearchUtils.getEngines()) { dropzones.appendChild(this.convertFromSearchEngine(engine)); } // 置換 let enumerator = Services.wm.getEnumerator('navigator:browser'); while (enumerator.hasMoreElements()) { let ul = enumerator.getNext().document.querySelector('#' + DragAndDropZonesPlus.ID + ' ul'); // クリア while (ul.hasChildNodes()) { ul.firstChild.remove(); } ul.appendChild(dropzones.cloneNode(true)); } }, /** * ドロップゾーンを削除する。 */ remove: function () { this.wrapper.remove(); }, /** * prefs.jsの設定値を元に、ドロップゾーンを作成する。 * @param {SearchEngine} engine * @returns {HTMLLIElement} */ convertFromSearchEngine: function (engine) { let li = document.createElementNS(DOMUtils.HTML_NS, 'li'); // インデックス li.dataset.engineIndex = engine.index; // dropzone属性 let dropzone = DOMUtils.getAttributeAsDOMTokenList(li, 'dropzone'); dropzone.add('link'); if (engine.accept === 'text/plain') { dropzone.add('string:text/plain'); if (engine.method === 'POST') { dropzone.add('file:text/*'); } } else { dropzone.add('file:' + engine.accept); } li.setAttribute('dropzone', dropzone); // アイコン let icon = new Image(16, 16); icon.src = engine.icon || DropzoneUtils.DEFAULT_ICON; li.appendChild(icon); // 表示名 li.appendChild(new Text(engine.name)); li.dataset.name = engine.name; return li; }, /** * windowに追加するイベントリスナーが補足するイベントの種類。 * @type {string[]} * @access protected */ eventTypesForWindow: ['dragstart', 'dragover', 'dragenter', 'dragleave', 'dragend', 'drop'], /** * ドラッグ中のアイテムの種類。 * ドラッグ中でなければnull。 * @type {?string[]} * @access protected */ itemTypesDuringDrag: null, /** * dragstartイベント。 * ウィンドウ外からドラッグしている場合はnull。 * @type {?DragEvent} * @access protected */ dragstartEvent: null, /** * ドラッグ開始後、dragoverイベントが既に発生していれば真。 * @type {booelan} * @access protected */ dragoverEventAlreadyFired: true, /** * ドラッグしようとしているアイテムの種類を取得する。 * @param {DragEvent} event - dragstartイベント。 * @returns {string[]} * @access protected */ getItemTypes: function (event) { let types = []; let target = event.target; let name = target.localName || target.nodeName; if (target.ownerDocument === document && event.dataTransfer.getData('text/plain') || ['a', 'img', '#text'].indexOf(name) !== -1 || ['input', 'textarea'].indexOf(name) !== -1 && !target.draggable) { // ロケーションバーや検索窓からのドラッグ、 // またはソースノードがリンク・画像・文字列、ドラッグ不可のテキスト入力欄なら types.push('string:text/plain'); } if (name === 'img' || name === 'a' && target.getElementsByTagName('img')[0]) { // ソースノードが画像、または画像を含むリンクなら types.push('file:image/*'); } return types; }, /** * drop-active-validクラスが付いた要素を返す。 * @returns {?HTMLLIElement} * @access protected */ getActiveValidDropzone: function () { return this.wrapper.getElementsByClassName('drop-active-valid')[0]; }, /** * dragstartイベントから、対象の文字列を取得する。 * @param {DragEvent} event * @returns {string} * @access protected */ getTextFromDragstartEvent: function (event) { let text = ''; let selection; let selectedString = ''; let target = event.target; let localName = target.localName; let doc = target.ownerDocument; if ('getSelection' in doc) { selection = doc.getSelection(); if (selection) { selectedString = selection.toString(); if (selectedString && (localName === 'a' || target.nodeType === Node.TEXT_NODE)) { // リンクか選択範囲をドラッグしていれば let x = event.clientX, y = event.clientY; if (!DOMUtils.isSuperposedCoordinateOnSelection(selection, x, y)) { // ドラッグ開始位置が選択範囲外なら let element = doc.elementFromPoint(x, y); if (element && (element.localName === 'a' || (element = DOMUtils.getParentElementByTagName(element, 'a')))) { // 選択範囲が重なったリンクの、選択範囲でない部分をドラッグしていれば text = gatherTextUnder(element); } } } } } if (!text && ['a', 'img'].indexOf(localName) !== -1) { // リンクか画像をドラッグしていれば if (selectedString && selection && selection.containsNode(target, true)) { // ソースノードが選択範囲と重なっており、 // リンクの一部分だけが選択されている場合は、選択範囲とドラッグ開始位置が重なっていれば text = selectedString; } if (!text) { if (localName === 'img') { // 画像をドラッグしていれば text = selectedString || target.alt || target.title; if (!text) { let figcaption = DOMUtils.getFigcaption(target); if (figcaption) { text = gatherTextUnder(figcaption); } } } else { // リンクをドラッグしていれば text = gatherTextUnder(target); } } } return text.trim() || event.dataTransfer.getData('text/plain').trim(); }, /** * ウィンドウ外からドロップされた文字列情報を取得する。 * @param {DragEvent} event - dropイベント。 * @prams {boolean} [forceString] - 真が指定されていれば、常にFileインスタンスの代わりにファイル名を返す。 * @returns {?(string|File)} * @access protected */ getTextFromDropEvent: function (event, forceString = false) { let dropFile = null, dropText = ''; let files = event.dataTransfer.files; if (files.length > 0) { // ファイルをドロップしていれば if (!forceString) { for (let file of files) { if (mimeTypeIsTextBased(file.type)) { // テキストファイルなら dropFile = file; break; } } } if (!dropFile) { // テキスト形式でないファイルがドロップされているかforceStringが指定されていれば、ファイル名を取得する dropText = files[0].name; } } else { dropText = event.dataTransfer.getData('text/plain'); } return dropFile ? dropFile : dropText.trim() || null; }, /** * dragstartイベントから、対象の画像URLを取得する。 * @param {DragEvent} event * @returns {?string} * @access protected */ getImageURLFromDragstartEvent: function (event) { let url = null; let target = event.target; switch (target.localName) { case 'img': url = target.src; break; case 'a': let images = target.getElementsByTagName('img'); if (images.length === 1) { url = images[0].src; } else { let image = target.ownerDocument.elementFromPoint(event.clientX, event.clientY); url = image.localName === 'img' && target.contains(image) ? image.src : images[0].src; } break; } return url; }, /** * URLからファイルを取得する。 * @param {string} url - ファイルのURL。 * @param {Function} [callback] - 第1引数に{@link Ci.nsIMIMEInputStream}。 * @access protected */ getMIMEInputStreamFromURL: function (url, callback) { let channel = NetUtil.newChannel(url); channel.loadFlags |= Ci.nsIRequest.LOAD_FROM_CACHE; let stream = channel.open(); let mimeStream = new MIMEInputStream(); mimeStream.addHeader('content-type', channel.contentType); mimeStream.setData(stream); callback(mimeStream); }, /** * ドロップされたデータを、ドロップゾーンに結びつけられたエンジンで検索する。 * @param {(string|Blob|nsIMIMEInputStream)} data - 検索する文字列、またはファイル。 * @param {number} engineIndex - prefs.jsに保存されているインデックス。 * @param {DragEvent} event - どのキーを押しているか取得するためのdropイベント。 * @access protected */ searchDropData: function (data, engineIndex, event) { let mimeType = data.type; if (mimeType && mimeTypeIsTextBased(mimeType) && !/^(?:image|audio)\//.test(mimeType)) { // ドロップされたデータがテキストファイルなら、文字列に変換しておく let fileReader = new FileReader(); fileReader.addEventListener('load', () => { this.searchDropData(fileReader.result, engineIndex, event); }); fileReader.readAsText(data); return; } let engine = SearchUtils.getEngineByIndex(engineIndex); if (engine) { if (engine.browserSearchEngine) { // ブラウザの検索窓のエンジンなら let browserSearchEngine = Services.search.getEngineByName(engine.name); if (browserSearchEngine) { let submission = browserSearchEngine.getSubmission(data); this.openSearchResult(submission.uri.spec, event, submission.postData); } } else { // ユーザー定義のエンジンなら if (engine.method === 'POST') { for (let i = 0, l = engine.params.length; i < l; i++) { if (engine.params[i][1].includes('{searchTerms}')) { engine.params[i][1] = data; } } StringUtils.encodeMultipartFormData(engine.params, postData => { this.openSearchResult(engine.url, event, postData); }, engine.encoding); } else { let encodedString; try { encodedString = TextToSubURI.ConvertAndEscape(engine.encoding, data); } catch (e) { if (e.result === Cr.NS_ERROR_UCONV_NOCONV) { encodedString = TextToSubURI.ConvertAndEscape(StringUtils.THE_ENCODING, data); } else { throw e; } } this.openSearchResult(engine.url.replace(/{searchTerms}/g, encodedString), event); } } } }, /** * ユーザー設定に基づき、適切な場所で検索結果を開く。 * @param {string} url * @param {DragEvent} event - どのキーを押しているか取得するためのdropイベント。 * @param {nsIInputStream} [postData] * @access protected */ openSearchResult: function (url, event, postData = null) { let where = Preferences.get(this.ROOT_BRANCH_NAME + 'where', 'tab'); if (where === 'current') { openUILink(url, event, { postData: postData }); } else { openUILinkIn(url, where, { postData: postData }); } }, }; /** * 検索窓のエンジンを追加・削除したときのオブザーバ。 */ let BrowserSearchEngineModifiedObserver = { /** * 監視する項目。 * @constant {string} */ SEARCH_ENGINE_TOPIC: 'browser-search-engine-modified', /** * 検索エンジンが削除されたときの通知。 * @constant {string} */ SEARCH_ENGINE_REMOVED: 'engine-removed', /** * 検索エンジンの情報が変更されたときの通知。 * @constant {string} */ SEARCH_ENGINE_CHNAGED: 'engine-changed', /** * 検索エンジンが追加されたときの通知。 * @constant {string} */ SEARCH_ENGINE_ADDED: 'engine-added', /** * オブザーバを追加する。 * ブラウザ起動時に呼び出し、新しいウィンドウが開かれたときは呼び出さない。 */ init: function () { Services.obs.addObserver(this, this.SEARCH_ENGINE_TOPIC, false); Services.obs.addObserver(this, DragAndDropZonesPlus.ID, false); }, /** * オブザーバを削除する。 */ stop: function () { Services.obs.removeObserver(this, this.SEARCH_ENGINE_TOPIC); Services.obs.removeObserver(this, DragAndDropZonesPlus.ID); }, /** * 通知を受け取るメソッド。 * @param {*} subject * @param {string} topic * @param {string} data */ observe: function (subject, topic, data) { switch (topic) { case this.SEARCH_ENGINE_TOPIC: let browserEngine = subject.QueryInterface(Ci.nsISearchEngine); switch (data) { case this.SEARCH_ENGINE_ADDED: // 検索窓にエンジンが追加されたとき if (Preferences.get(SettingsUtils.ROOT_BRANCH_NAME + 'automaticallyReflect', true)) { // ドロップゾーンの自動追加が有効なら let engine = SearchUtils.convertEngineFromBrowser(browserEngine); engine.index = SearchUtils.getDropzoneLength(); SearchUtils.setEngine(engine.index, engine); DropzoneUtils.update(); } break; case this.SEARCH_ENGINE_REMOVED: // 検索窓からエンジンが削除されたとき for (let i = 0, l = SearchUtils.getDropzoneLength(); i < l; i++) { let branch = new Preferences(SettingsUtils.ROOT_BRANCH_NAME + 'engines.' + i + '.'); if (branch.get('name') === browserEngine.name && !branch.get('url')) { // 同名のエンジン、かつユーザー定義エンジンでなければ gPrefService.getBranch(SettingsUtils.ROOT_BRANCH_NAME + 'engines.').deleteBranch(i + '.'); DropzoneUtils.update(); break; } } break; } break; case DragAndDropZonesPlus.ID: switch (data) { case UninstallObserver.TYPE: // アンインストール時 this.stop(); break; } break; } }, }; /** * 設定画面。 */ let SettingsScreen = { /** * XUL名前空間。 * @constant {string} */ XUL_NS: 'http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul', /** * Base64に変換したとき、データが何倍になるか。 * @constant {number} */ BASE64_SIZE_RATIO: 4 / 3, /** * 設定画面タブのアイコン。 * @constant {string} */ ICON: '', /** * メニューバーの「ツール」に、設定画面を開くオプションを追加。 */ addToMenu: function () { let menuItem = document.createElementNS(this.XUL_NS, 'menuitem'); menuItem.id = DragAndDropZonesPlus.ID + '-menuitem'; menuItem.setAttribute('label', DragAndDropZonesPlus.NAME); menuItem.setAttribute('image', this.ICON); menuItem.classList.add('menuitem-iconic'); menuItem.addEventListener('command', () => this.open()); document.getElementById('menu_ToolsPopup').appendChild(menuItem); }, /** * 設定画面を開く。 */ open: function () { if (this.tab) { // すでに同じウィンドウで開いていれば gBrowser.selectedTab = this.tab; } else { let enumerator = Services.wm.getEnumerator('navigator:browser'); while (enumerator.hasMoreElements()) { let win = enumerator.getNext(); if (win[DragAndDropZonesPlus.ID + '_settingsTab']) { // すでに別ウィンドウで開いていれば win.focus(); win.gBrowser.selectedTab = win[DragAndDropZonesPlus.ID + '_settingsTab']; return; } } this.tab = gBrowser.addTab('about:blank'); gBrowser.getBrowserForTab(this.tab).addEventListener('load', this, true); gBrowser.selectedTab = this.tab; gBrowser.tabContainer.addEventListener('TabClose', this); } }, /** * イベントハンドラ。 * @param {Event} event */ handleEvent: function (event) { let target = event.target; switch (event.type) { case 'load': let browser = gBrowser.getBrowserForTab(this.tab); if (browser.documentURI.spec === 'about:blank') { this.show(); } else { // 同じタブで別のページが開かれたら browser.removeEventListener(event.type, this, event.eventPhase === Event.CAPTURING_PHASE); this.tab = null; return; } break; case 'TabClose': let closedTab = event.target; if (closedTab === this.tab) { gBrowser.tabContainer.removeEventListener(event.type, this); this.tab = null; } break; case 'beforeunload': delete window[DragAndDropZonesPlus.ID + '_settingsTab']; break; case 'submit': // OKボタン target.querySelector('[type=submit],button:not([type])').disable = true; event.preventDefault(); let engines = []; for (let row of target.getElementsByTagName('tbody')[0].rows) { let engine = {}; let name = row.querySelector('[name=name]'); if (name.readOnly) { // 検索窓のエンジンなら engine.browserSearchEngine = true; } engine.name = name.value; if (!engine.browserSearchEngine) { engine.url = row.querySelector('[name=url]').value; } if (engine.name && (engine.browserSearchEngine ? Services.search.getEngineByName(engine.name) : engine.url)) { // 検索エンジン名が空文字列でなければ、 // かつブラウザのエンジンであればそれが存在するなら、ユーザー定義エンジンであればURLが空文字列でなければ if (!engine.browserSearchEngine) { // ユーザー定義エンジンであれば let icon = row.querySelector('[name=icon]').value; if (icon) { engine.icon = icon; } engine.method = row.querySelector('[name=method]').value; if (engine.method === 'GET') { // GETメソッドなら if (!engine.url.includes('{searchTerms}')) { engine.url += '{searchTerms}'; } } else { // POSTメソッドなら engine.params = []; for (let pair of row.querySelectorAll('tbody > tr')) { let name = pair.querySelector('[name=post-param-name]').value; let value = pair.querySelector('[name=post-param-value]').value; if (name || value) { // 名前と値どちらかが入力されていれば engine.params.push([name, value]); } } } engine.accept = row.querySelector('[name=accept]').value; engine.encoding = row.querySelector('[name=encoding]').value; } engines.push(engine); } } let prefBranch = gPrefService.getBranch(SettingsUtils.ROOT_BRANCH_NAME); let where = target.where.value; if (where === 'tab') { prefBranch.clearUserPref('where'); } else { Preferences.set(SettingsUtils.ROOT_BRANCH_NAME + 'where', where); } if (target['automatically-reflect'].checked) { prefBranch.clearUserPref('automaticallyReflect'); } else { Preferences.set(SettingsUtils.ROOT_BRANCH_NAME + 'automaticallyReflect', false); } SearchUtils.setEngines(engines); DropzoneUtils.update(); this.close(); break; case 'contextmenu': // 選択された要素を取得しておく this.selectedElement = target; case 'click': switch (target.name) { case 'icon': // 選択された要素を取得しておく this.selectedElement = target; if (target.type !== 'menu') { // button[type=menu] の代替 // Bug 897102 – Update <menu> to spec <https://bugzilla.mozilla.org/show_bug.cgi?id=897102> event.preventDefault(); let doc = gBrowser.getBrowserForTab(this.tab).contentDocument; let menu = doc.getElementById('icon-menu'); let iconMenupopup = document.createElementNS(this.XUL_NS, 'menupopup'); for (let menuitem of menu.getElementsByTagName('menuitem')) { let xulMenuitem = document.createElementNS(this.XUL_NS, 'menuitem'); xulMenuitem.setAttribute('label', menuitem.label); xulMenuitem.setAttribute('value', menuitem.id); iconMenupopup.appendChild(xulMenuitem); } iconMenupopup.addEventListener('click', function (event) { doc.getElementById(event.target.value).click(); }); iconMenupopup.addEventListener('popuphidden', function (event) { event.currentTarget.remove(); }); document.documentElement.appendChild(iconMenupopup).openPopup(target); } break; case 'delete': // 行の削除 this.deleteEngine(target); break; case 'add-row': // 行の追加 this.insertEmptyRow(target); break; case 'params': // POSTパラメータの開閉 DOMUtils.getParentElementByTagName(target, 'tr').classList.toggle('displaying-post-params'); break; case 'export': // エクスポート SettingsUtils.exportToFile(gBrowser.getBrowserForTab(this.tab).contentDocument.defaultView, filePath => { showPopupNotification(_('%s へ設定をエクスポートしました。').replace('%s', filePath), this.tab); }); break; case 'import': // インポート SettingsUtils.importFromFile(gBrowser.getBrowserForTab(this.tab).contentDocument.defaultView, (fileName, errorMessage) => { if (errorMessage) { showPopupNotification(_('%s からのインポートに失敗しました。').replace('%s', fileName) + '\n' + errorMessage, this.tab, 'warning'); } else { gBrowser.getBrowserForTab(this.tab).contentDocument.defaultView.location.reload(); showPopupNotification(_('%s からのインポートが完了しました。').replace('%s', fileName), this.tab); } }); break; case 'additional-import': // 追加インポート this.addEnginesFromFile((fileName, errorMessage) => { if (errorMessage) { showPopupNotification(_('%s からのインポートに失敗しました。').replace('%s', fileName) + '\n' + errorMessage, this.tab, 'warning'); } else { showPopupNotification(_('%s からのインポートが完了しました。').replace('%s', fileName) + '\n' + _('インポートした設定を保存するには、「OK」ボタンをクリックしてください。'), this.tab); } }); break; case 'import-from-text': // JSON文字列から追加インポート Version1Settings.getEnginesFromText(gBrowser.getBrowserForTab(this.tab).contentDocument.defaultView, (engines, errorMessage) => { if (errorMessage) { showPopupNotification(_('JSON文字列からのインポートに失敗しました。') + '\n' + errorMessage, this.tab, 'warning'); } else { this.addEngines(engines); showPopupNotification(_('JSON文字列からのインポートが完了しました。') + '\n' + _('インポートした設定を保存するには、「OK」ボタンをクリックしてください。'), this.tab); } }); break; case 'cancel': // キャンセル this.close(); break; case 'get-icons': // 未取得アイコンの一括取得 target.disabled = true; this.setIconsToEngineWithout(() => { target.disabled = false; showPopupNotification(_('アイコンの取得が完了しました。'), this.tab); }); break; case 'initialize': case 'uninstall': // 設定の初期化、またはすべての設定の削除 if (gBrowser.getBrowserForTab(this.tab).contentDocument.defaultView.confirm(_('本当に、『%s』のすべての設定を削除してもよろしいですか?').replace('%s', DragAndDropZonesPlus.NAME))) { if (target.name === 'initialize') { DragAndDropZonesPlus.initialize(); gBrowser.getBrowserForTab(this.tab).contentDocument.defaultView.location.reload(); showPopupNotification(_('設定の初期化が完了しました。'), this.tab); } else { this.close(); DragAndDropZonesPlus.uninstall(); showPopupNotification(_('設定の削除が完了しました。当スクリプト自体を削除しなければ、次回のブラウザ起動時にまた設定が作成されます。'), gBrowser.selectedTab); } } break; default: let parent = target.parentElement; if (parent) { switch (parent.id) { case 'row-contextmenu': // 行のコンテキストメニュー // 行の挿入 this.insertEmptyRow(this.selectedElement, target.id === 'add-row-above'); break; case 'icon-menu': // アイコンのメニュー let url; switch (target.id) { case 'set-icon-from-local-file': IconUtils.getFromLocalFile(gBrowser.getBrowserForTab(this.tab).contentDocument.defaultView, (dataURL, errorMessage) => { if (dataURL) { this.showIcon(dataURL); } else { showPopupNotification(errorMessage, this.tab, 'warning'); } }); break; case 'set-icon-from-url': // 入力ダイアログを開く url = gBrowser.getBrowserForTab(this.tab).contentDocument.defaultView.prompt(_('WebページのURL、または画像ファイルのURLを入力してください。')); if (url) { IconUtils.getFromUrl(url, (dataURL, errorMessage) => { if (dataURL) { this.showIcon(dataURL); } else { showPopupNotification(errorMessage, this.tab, 'warning'); } }); } break; case 'set-icon-from-clipboard': // クリップボードのデータを取得する IconUtils.getIconFromClipboard((dataURL, errorMessage) => { if (dataURL) { this.showIcon(dataURL); } else { showPopupNotification(errorMessage, this.tab, 'warning'); } }); break; case 'restore-default-icon': // Faviconを取得し設定 url = DOMUtils.getParentElementByTagName(this.selectedElement, 'tr').querySelector('[name=url]').value; if (url) { IconUtils.getFaviconFromSiteUrl(url, (dataURL) => { if (dataURL) { this.showIcon(dataURL); } else { this.selectedElement.value = ''; this.selectedElement.firstElementChild.src = DropzoneUtils.DEFAULT_ICON; } }); } else { this.selectedElement.value = ''; this.selectedElement.firstElementChild.src = DropzoneUtils.DEFAULT_ICON; } break; } break; } } } break; case 'keypress': let key = event.key; switch (key) { case 'Enter': // Enterキーが押されたとき if (target.matches('tbody tr, tbody tr *')) { // 行内でキーが押されたとき let shiftKey = event.getModifierState('Shift'); if (shiftKey || event.getModifierState('Alt') || event.getModifierState('Control')) { // Shiftキー、Altキー、Ctrlキーいずれかが押されていれば event.preventDefault(); // 行を追加する this.insertEmptyRow(target, shiftKey, true); } } break; case 'ArrowUp': case 'ArrowDown': // 上矢印キー、または下矢印キーが押されたとき if (target.localName === 'input' && (event.getModifierState('Alt') || event.getModifierState('Control'))) { // input要素上でAltキーかCtrlキーが押されていれば let current = DOMUtils.getParentElementByTagName(target, 'tr'); let currentIndex = current.sectionRowIndex + 1; let indexes = key === 'ArrowUp' ? '-n+' + (currentIndex - 1) : 'n+' + (currentIndex + 1); let rows = current.parentElement.querySelectorAll(':not(td) > table > tbody > tr:nth-of-type(' + indexes + '):not(.browser-search-engine)'); let sibling = key === 'ArrowUp' ? rows[rows.length - 1] : rows[0]; if (sibling) { event.preventDefault(); sibling.querySelector('[name=' + target.name + ']').focus(); } } break; } break; case 'change': switch (target.name) { case 'method': // メソッドが変更されたとき let row = DOMUtils.getParentElementByTagName(target, 'tr'); if (target.value === 'POST') { // POSTメソッド row.classList.add('post'); } else { // GETメソッド row.querySelector('[name=accept]').value = 'text/plain'; row.classList.remove('displaying-post-params', 'post'); } break; case 'url': let url = target.value; if (url) { // URLのバリデート let validationMessage = ''; try { NetUtil.newURI(url).QueryInterface(Ci.nsIURL); } catch (e) { if (e.result === Cr.NS_ERROR_MALFORMED_URI || e.result === Cr.NS_NOINTERFACE) { // 妥当なURLでなければ validationMessage = _('http:// などで始まるURLを入力してください。'); } else { throw e; } } if (target.validationMessage !== validationMessage) { target.setCustomValidity(validationMessage); } } break; case 'add-browser-engine': // ブラウザの検索エンジンの追加 this.insertEngine(JSON.parse(target.selectedOptions[0].dataset.engine)); target[0].selected = true; break; } break; // 行の並び替え case 'dragstart': if (target.matches('[draggable], [draggable] *')) { let row = DOMUtils.getParentElementByTagName(target, 'tr'), dataTransfer = event.dataTransfer; dataTransfer.setDragImage(row, 0, 0); dataTransfer.setData('application/x-sectionrowindex', row.sectionRowIndex); this.duringRowDrag = true; } break; case 'dragover': if (this.duringRowDrag) { let row = DOMUtils.getParentElementByTagName(target, 'tr'); if (row) { event.preventDefault(); this.resetDropzoneClasses(); switch (row.parentElement.localName) { case 'thead': DOMUtils.getParentElementByTagName(target, 'table').tBodies[0].rows[0].classList.add('active-dropzone-above'); break; case 'tbody': let rect = row.getBoundingClientRect(); if (event.clientY - rect.top < rect.height / 2) { // ポインタが行の真ん中より上にあれば row.classList.add('active-dropzone-above'); } else { // ポインタが行の真ん中より下にあれば row.classList.add('active-dropzone-below'); } break; case 'tfoot': let tbody = DOMUtils.getParentElementByTagName(target, 'table').tBodies[0]; tbody.rows[tbody.rows.length - 1].classList.add('active-dropzone-below'); break; } } } break; case 'dragleave': if (this.duringRowDrag && !target.ownerDocument.getElementsByTagName('table')[0].contains(event.relatedTarget)) { this.resetDropzoneClasses(); } break; case 'drop': if (this.duringRowDrag) { let doc = gBrowser.getBrowserForTab(this.tab).contentDocument; let refChild = doc.getElementsByClassName('active-dropzone-above')[0]; let tbody; if (refChild) { tbody = refChild.parentElement; } else { let targetRow = doc.getElementsByClassName('active-dropzone-below')[0]; tbody = targetRow.parentElement; if (targetRow) { refChild = targetRow.nextElementSibling; } else { return; } } tbody.insertBefore(tbody.rows[event.dataTransfer.getData('application/x-sectionrowindex')], refChild); this.resetDropzones(); } break; case 'dragend': if (this.duringRowDrag) { this.resetDropzones(); } break; } }, /** * 行の並べ替え終了時の処理。 */ resetDropzones: function () { this.duringRowDrag = false; this.resetDropzoneClasses(); }, /** * 行の並べ替えに関するクラス名の削除。 */ resetDropzoneClasses: function () { for (let row of gBrowser.getBrowserForTab(this.tab).contentDocument.querySelectorAll('.active-dropzone-above, .active-dropzone-below')) { row.classList.remove('active-dropzone-above', 'active-dropzone-below'); } }, /** * アイコンが設定されていない行について、アイコンを一括取得し設定する。 * @param {Function} [callback] * @access protected */ setIconsToEngineWithout: function (callback = function () { }) { let emptyIcons = gBrowser.getBrowserForTab(this.tab).contentDocument.querySelectorAll('tbody > tr [name=icon]:not([value]), tbody > tr [name=icon][value=""]'); // 進行状況 let progress = doc.createElement('progress'); progress.max = emptyIcons.length; progress.value = 0; target.parentElement.replaceChild(progress, target); (function getFavicon() { let icon = emptyIcons[progress.value]; if (icon) { let url = DOMUtils.getParentElementByTagName(icon, 'tr').querySelector('[name=url]').value; if (url) { IconUtils.getFaviconFromSiteUrl(url, function (dataURL) { icon.firstElementChild.src = icon.value = dataURL; progress.value = Number(progress.value) + 1; getFavicon(); }, function () { progress.value = Number(progress.value) + 1; getFavicon(); }); } else { progress.value = Number(progress.value) + 1; getFavicon(); } } else { // すべてのアイコンを取得し終えたら progress.parentElement.replaceChild(target, progress); callback(); } })(); }, /** * 行を削除する。 * @param {HTMLButtonElement} deleteButton - 削除ボタン。 * @access protected */ deleteEngine: function (deleteButton) { let row = DOMUtils.getParentElementByTagName(deleteButton, 'tr'); row.remove(); let name = row.querySelector('[name=name]'); if (name.readOnly) { // 検索窓の検索エンジンを削除していれば、検索窓の検索エンジンを追加するセレクトボックスで選択可能に deleteButton.ownerDocument.querySelector('[name=add-browser-engine] [label=' + StringUtils.quote(name.value) + ']').hidden = false; } }, /** * 指定した位置に空行を作成。 * @param {HTMLElement} target - 挿入位置の基準になる行に含まれる要素、または行を追加するtable要素。 * @param {boolean} [insertingBefore=false] - 基準になる行の上に挿入するならtrue。 * @param {boolean} [focus=false] - 新しい行にフォーカスを移すならtrue。 */ insertEmptyRow: function(target, insertingBefore = false, focus = false) { let table = target.localName === 'table' ? target : DOMUtils.getParentElementByTagName(target, 'table'); let targetRow = null; if (target.localName !== 'table' && target.name !== 'add-row') { let row = DOMUtils.getParentElementByTagName(target, 'tr'); targetRow = insertingBefore ? row : row.nextElementSibling; } let row = table.parentElement.localName === 'form' ? this.insertEngine(null, targetRow) : table.tBodies[0].insertBefore(table.getElementsByTagName('template')[0].content.firstChild.cloneNode(true), targetRow); if (focus && /^(?:input|select)$/.test(target.localName)) { // 追加した行にフォーカスを移す row.querySelector('[name=' + event.target.name + ']').focus(); } }, /** * 指定された位置に行を挿入する。 * @param {SearchEngine} [engine] - 指定しなかった場合は空行を挿入する。 * @param {HTMLTableRowElement} [child] - 指定しなかった場合は表本体の末尾に追加する。 * @return {?HTMLTableRowElement} ブラウザの検索エンジンが壊れたURLを保持していた場合は行を挿入しない。 */ insertEngine: function (engine = null, child = null) { let doc = gBrowser.getBrowserForTab(this.tab).contentDocument; let tbody = doc.getElementsByTagName('tbody')[0]; let row = tbody.getElementsByTagName('template')[0].content.firstChild.cloneNode(true); let paramsTbody = row.getElementsByTagName('tbody')[0]; let paramTemplate = paramsTbody.getElementsByTagName('template')[0].content.firstChild; if (!engine || engine.method === 'GET') { // GETメソッド let row = paramTemplate.cloneNode(true); row.querySelector('[name=post-param-value]').value = '{searchTerms}'; paramsTbody.appendChild(row); } if (engine) { if (engine.browserSearchEngine) { // ブラウザの検索エンジンなら row.classList.add('browser-search-engine'); doc.querySelector('[name=add-browser-engine] [label=' + StringUtils.quote(engine.name) + ']').hidden = true; } else { if (engine.method === 'POST') { // POSTメソッド row.classList.add('post'); } } for (let input of row.querySelectorAll('[name]')) { let name = input.name; if (engine[name]) { if (name === 'params') { // POSTパラメータなら if (!engine.browserSearchEngine) { let params = engine[name] || []; if (engine.browserSearchEngine) { // ブラウザの検索エンジンなら row.getElementsByTagName('tfoot')[0].remove(); } else { params = params.concat([['', '']]); } params.forEach(function ([name, value]) { let row = paramTemplate.cloneNode(true); let nameInput = row.querySelector('[name=post-param-name]'); let valueInput = row.querySelector('[name=post-param-value]'); nameInput.value = name; valueInput.value = value; if (engine.browserSearchEngine) { // ブラウザの検索エンジンなら nameInput.readOnly = true; valueInput.readOnly = true; row.querySelector('[name=delete]').remove(); } paramsTbody.appendChild(row); }); } } else { if (engine[name]) { input.value = engine[name]; } if (engine.browserSearchEngine) { // ブラウザの検索エンジンなら switch (name) { case 'name': input.readOnly = true; break; case 'url': input.readOnly = true; if (engine.method === 'POST') { let url; try { url = NetUtil.newURI(engine[name]).QueryInterface(Ci.nsIURL); } catch (e) { if (e.result === Cr.NS_ERROR_MALFORMED_URI || e.result === Cr.NS_NOINTERFACE) { // 妥当なURLでなければ return null; } else { throw e; } } let searchParams = new URLSearchParams(url.query); for (let [name, value] of engine.params) { searchParams.append(name, value); } url.query = searchParams.toString().replace(/%7BsearchTerms%7D/g, '{searchTerms}'); input.value = url.spec; } break; case 'icon': let cell = DOMUtils.getParentElementByTagName(input, 'td'); let img = input.firstElementChild; cell.replaceChild(img, input); img.src = engine[name] || DropzoneUtils.DEFAULT_ICON; break; default: let value = input.localName === 'select' ? input.selectedOptions[0].text : input.value; DOMUtils.getParentElementByTagName(input, 'td').textContent = value; } } switch (name) { case 'icon': if (!engine.browserSearchEngine && engine[name]) { input.firstElementChild.src = engine[name]; } break; case 'name': input.dataset.value = engine[name]; break; } } } } } if (engine && !child) { // 空行の挿入ではない、かつ末尾への追加なら let rows = tbody.rows; // 末尾の空行ではない行を取得 let previousRowSibling = Array.prototype.slice.call(rows).reverse().find(row => row.querySelector('[name=name]').value.trim() !== '' || row.querySelector('[name=url]').value.trim() !== ''); // 追加位置を取得 child = previousRowSibling ? previousRowSibling.nextElementSibling : rows[0]; } return tbody.insertBefore(row, child); }, /** * 行をドラッグ中ならtrue。 * @type {boolean} * @access protected */ duringRowDrag: false, /** * イベントが発生したタブに設定画面を描画する。 * @access protected */ show: function () { let doc = gBrowser.getBrowserForTab(this.tab).contentDocument, win = doc.defaultView; window[DragAndDropZonesPlus.ID + '_settingsTab'] = this.tab; this.printStatic(); // ブラウザの検索エンジンを追加するセレクトボックス let addingBrowserEngine = doc.getElementsByName('add-browser-engine')[0]; for (let engine of SearchUtils.getBrowserEngines()) { let option = new Option(engine.name); option.label = engine.name; option.dataset.engine = JSON.stringify(engine); addingBrowserEngine.add(option); } let table = doc.getElementsByTagName('table')[0]; let tbody = table.tBodies[0]; doc.addEventListener('click', this); tbody.addEventListener('contextmenu', this); tbody.addEventListener('keypress', this); doc.addEventListener('submit', this); doc.addEventListener('change', this); win.addEventListener('beforeunload', this); tbody.addEventListener('dragstart', this); table.addEventListener('dragover', this); table.addEventListener('dragleave', this); table.addEventListener('drop', this); win.addEventListener('dragend', this); // 設定を表示 for (let engine of SearchUtils.getEngines()) { this.insertEngine(engine); } // 空行を追加 this.insertEngine(); }, /** * 表に検索エンジンを追加する。 * @param {SearchEngine[]} engines * @access protected */ addEngines: function (engines) { let doc = gBrowser.getBrowserForTab(this.tab).contentDocument; let addingBrowserEngine = doc.getElementsByName('add-browser-engine')[0]; for (let engine of engines) { let option = addingBrowserEngine.querySelector('[label=' + StringUtils.quote(engine.name) + ']'); if (option) { // 同名の検索エンジンがブラウザに存在すれば if (!option.hidden) { // それが追加されていないエンジンなら this.insertEngine(JSON.parse(option.dataset.engine)); option.hidden = true; } } else { this.insertEngine(engine); } } }, /** * 取得したアイコンをアイコン設定ボタンに表示する。 * @param {string} dataURL - アイコンのDataURL * @access protected */ showIcon: function (dataURL) { if (dataURL.length > SettingsUtils.MAX_PREFERENCE_VALUE_LENGTH) { showPopupNotification(_('アイコンの設定に失敗しました。約 %s KiB までの画像を設定できます。').replace('%s', SettingsUtils.MAX_PREFERENCE_VALUE_LENGTH / this.BASE64_SIZE_RATIO / 1024), this.tab, 'warning'); } else { this.selectedElement.firstElementChild.src = this.selectedElement.value = dataURL; } }, /** * ファイルから設定を追加する。 * @param {Function} [callback] - 第1引数にファイル名。インポートに失敗していれば、第2引数にエラーメッセージ。 * @access protected */ addEnginesFromFile: function (callback = function () { }) { SearchUtils.getSearchEnginesFromFile(gBrowser.getBrowserForTab(this.tab).contentDocument.defaultView, (engines, settingsDocument, fileName, errorMessage) => { if (engines) { this.addEngines(engines); callback(fileName); } else { callback(fileName, errorMessage); } }); }, /** * 設定画面を閉じる。 * @access protected */ close: function () { if (gBrowser.tabs.length === 1) { // 設定画面以外のタブが存在しなければ let enumerator = Services.wm.getEnumerator('navigator:browser'); enumerator.getNext(); if (!enumerator.hasMoreElements()) { // 他にウィンドウが存在しなければ、ホームページに移動する let homePage = gHomeButton.getHomePage(); gBrowser.getBrowserForTab(this.tab).contentDocument.defaultView.location.assign( homePage === 'about:blank' ? 'about:home' : homePage ); } else { gBrowser.getBrowserForTab(this.tab).contentDocument.defaultView.close(); } } else { gBrowser.removeTab(this.tab); } }, /** * 設定画面の静的部分を描画。 * @access protected */ printStatic: function () { let doc = gBrowser.getBrowserForTab(this.tab).contentDocument; // Favicon let link = doc.createElement('link'); link.rel = 'icon'; link.href = this.ICON; doc.head.appendChild(link); // タイトル doc.title = DragAndDropZonesPlus.NAME; // スタイルシート let styleSheet = doc.head.appendChild(doc.createElement('style')).sheet; let cssRules = styleSheet.cssRules; // Bug 906353 – Add support for css4 selector :matches(), the standard of :-moz-any(). <https://bugzilla.mozilla.org/show_bug.cgi?id=906353> [ ':root {' + 'height: 100%;' + 'color: -moz-DialogText;' + 'background: -moz-Dialog;' + '}', // 行のドラッグ '[draggable=true],' + '[draggable=true] [readonly]:not([name=name]),' + 'td table [readonly] {' + 'cursor: move;' + '}', '[name=name] {' + 'width: 150px;' + '}', '[name=url] {' + 'width: 400px;' + '}', 'input:not([type]), [type=text], [type=url] {' + 'width: 100%;' + '}', // 検索窓のエンジン '.browser-search-engine {' + 'font: -moz-field;' + '}', '.browser-search-engine > :not(:first-child):not(:last-child) {' + 'padding-left: 8px;' + '}', '.browser-search-engine input {' + 'margin-left: -2px;' + 'background-color: transparent;' + 'border: none;' + 'text-overflow: ellipsis ellipsis;' + '}', // 行の背景色・枠線 'table {' + 'border-collapse: collapse;' + 'width: 100%;' + '}', 'thead th {' + '-moz-appearance: treeheadercell;' + 'font-weight: normal;' + '}', 'th, td {' + 'padding: 3px;' + '}', 'tbody > tr {' + 'background: whitesmoke;' + '}', 'tbody > tr:nth-child(2n) {' + 'background: gainsboro;' + '}', 'thead {' + 'border-top: solid 1px gray;' + 'border-left: solid 1px gray;' + 'border-right: solid 1px gray;' + '}', 'tbody {' + 'border-left: solid 1px gray;' + 'border-bottom: solid 1px gray;' + 'border-right: solid 1px gray;' + '}', // 行の追加ボタン 'tfoot td {' + 'padding: 0;' + '}', '[name=add-row]::before {' + 'content: url("");' + 'margin-right: 0.5em;' + 'vertical-align: -4px;' + '}', '[name=add-row] {' + 'border-top: none;' + 'border-left: solid 1px gray;' + 'border-bottom: solid 1px gray;' + 'border-right: solid 1px gray;' + 'border-radius: 0 0 0.2em 0.2em;' + 'background: linear-gradient(lightgrey, silver);' + 'position: relative;' + 'top: -1px;' + 'left: -1px;' + '}', '[name=add-row]:not([disabled]):-moz-any(:hover, :focus, :active) {' + 'background: gainsboro;' + '}', // キャンセル・OKボタン '#submit-buttons {' + 'text-align: right;' + '}', '#submit-buttons button {' + 'margin-left: 1em;' + '}', 'button:not([type]), button[type=submit] {' + 'font-size: 2em;' + '}', // その他の操作 ':not(td) > select {' + 'margin-left: 0.5em;' + '}', 'form > label {' + 'display: block;' + '}', // キーボードショートカット '#shortcut-keys ul {' + 'list-style: none;' + 'padding-left: 0;' + '}', '#shortcut-keys :not(kbd) > kbd:last-of-type::after {' + 'margin-right: 1em;' + 'content: ":";' + '}', 'kbd kbd {' + 'font-size: 0.75em;' + 'border-radius: 0.5em;' + 'border-style: solid;' + 'border-width: 0.15em 0.3em 0.45em;' + 'border-color: rgba(0,0,0,0.2) rgba(0,0,0,0.1) rgba(255,255,255,0.2);' + 'background-origin: border-box;' + 'background: gainsboro;' + 'color: black;' + 'display: inline-flex;' + 'align-items: center;' + '-moz-box-sizing: border-box;' + 'box-sizing: border-box;' + 'padding-left: 0.3em;' + 'width: 3em;' + 'height: 3em;' + 'margin-left: 0.2em;' + 'margin-right: 0.2em;' + '}', '.control {' + 'width: 4em;' + 'color: transparent;' + '}', '.control::before {' + 'content: "Ctrl";' + 'color: black;' + '}', '.shift {' + 'width: 5em;' + '}', '.shift::before {' + 'content: "⇧";' + 'margin-right: 0.2em;' + '}', '.alt {' + 'color: darkcyan;' + '}', '.enter {' + 'width: 4.5em;' + 'height: 5em;' + '}', '.enter::after {' + 'content: "⏎";' + '}', 'td {' + 'height: 100%;' // セル内で高さのパーセント指定が使えるように + '}', // アイコン 'td > img {' + 'display: block;' + 'margin-left: auto;' + 'margin-right: auto;' + '}', '[name=icon] {' + 'width: 100%;' + 'height: 100%;' + '}', // 行の削除ボタン '[name=delete] {' + 'border: none;' + 'padding: 0;' + 'background: transparent;' + 'width: 100%;' + 'height: 100%;' + 'overflow: hidden;' + 'border-radius: 0.5em;' + '}', '[name=delete]:not([disabled]):-moz-any(:hover, :focus, :active) {' + 'box-shadow: 1px 1px 2px rgba(0, 0, 0, 0.3) inset, -1px 0 2px rgba(0, 0, 0, 0.3) inset, 0 -1px 1px rgba(255, 255, 255, 0.3) inset;' + '}', '[name=delete]:not([disabled]):active {' + 'background-color: lightcoral;' + '}', 'tbody tr:only-of-type [name=delete] {' // 行が一つだけなら、削除ボタンは表示しない + 'visibility: hidden;' + '}', // Bug 895182 – [CSS Filters] Implement parsing for blur, brightness, contrast, grayscale, invert, opacity, saturate, sepia <https://bugzilla.mozilla.org/show_bug.cgi?id=895182> '[name=delete] img {' + 'background: url("");' + '}', '[name=delete]:active img {' + 'background: url("");' + '}', // 行の移動 '.active-dropzone-above {' + 'border-top: solid 0.5em lightblue;' + '}', '.active-dropzone-below {' + 'border-bottom: solid 0.5em lightblue;' + '}', // POSTパラメータの開閉ボタン 'td > div {' + 'display: flex;' + 'align-items: flex-start;' + '}', '[name=url] {' + 'flex-grow: 1;' + '}', '[name=params] {' + 'display: none;' + 'white-space: nowrap;' + 'border: 1px solid lightblue;' + 'box-shadow: 0px -2px 0px rgba(204, 223, 243, 0.3) inset, 0px 0px 1px rgba(0, 0, 0, 0.1);' + 'border-radius: 5px;' + 'background: linear-gradient(ghostwhite, aliceblue) ghostwhite;' + 'vertical-align: middle;' + '}', '[name=params]::after {' + 'content: "";' + 'display: inline-block;' + 'width: 20px;' + 'height: 20px;' + 'background: url("");' + 'vertical-align: middle;' + '}', '[name=params]:not([disabled]):-moz-any(:hover, :focus, :active)::after {' + 'background-position: 0 -64px;' + '}', // データの種類 '[name=accept] :not([value="text/plain"]) {' + 'display: none;' + '}', // POSTメソッドが選択されているとき '.post [name=url] {' + 'width: auto;' + '}', '.post [name=params] {' + 'display: inline-block;' + '}', '.post [name=accept] option {' + 'display: block;' + '}', // POSTパラメータ 'td table {' + 'display: none;' + 'border: 1px solid lightblue;' + 'border-collapse: separate;' + 'border-radius: 5px 0 5px 5px;' + 'background: linear-gradient(aliceblue, lavender) aliceblue;' + 'margin-top: -2px;' + '}', 'td table tr {' + 'background: transparent !important;' + '}', 'td table td {' + 'padding-left: 0;' + 'padding-right: 0;' + '}', 'td table [name=add-row] {' + 'background: linear-gradient(aliceblue, lavender);' + 'border: darkgray solid 1px;' + 'border-radius: 5px;' + '}', 'td table [name=add-row]:not([disabled]):-moz-any(:hover, :focus, :active) {' + 'background: aliceblue;' + '}', // POSTパラメータ展開時 '.displaying-post-params > * {' + 'vertical-align: top;' + '}', '.displaying-post-params > td > [name=delete] {' + 'height: 1.6em;' + '}', '.displaying-post-params [name=params] {' + 'border-bottom: none;' + 'border-bottom-left-radius: 0;' + 'border-bottom-right-radius: 0;' + 'box-shadow: none;' + '}', '[name=params]:not([disabled]):-moz-any(:hover, :focus, :active)::after {' + 'background-position: 0 -64px;' + '}', '.displaying-post-params [name=params]::after {' + 'background-position: 0 -128px;' + '}', '.displaying-post-params [name=params]:not([disabled]):-moz-any(:hover, :focus, :active)::after {' + 'background-position: 0 -192px;' + '}', '.displaying-post-params table {' + 'display: table;' + '}', ].forEach(function (rule) { styleSheet.insertRule(rule, cssRules.length); }); let cell, input, select, menu, button, img, div, label, section, kbd, key, li, dl; let form = doc.createElement('form'); let table = doc.createElement('table'); // 見出し let thead = table.createTHead(); let headRow = thead.insertRow(); // 本体 let tbody = table.createTBody(); let template = doc.createElement('template'); let row = template.content.appendChild(doc.createElement('tr')); // アイコン headRow.appendChild(doc.createElement('th')); cell = row.insertCell(); cell.draggable = true; input = doc.createElement('button'); input.type = 'menu'; input.name = 'icon'; input.title = _('アイコンを変更'); img = new Image(16, 16); img.src = DropzoneUtils.DEFAULT_ICON; img.alt = ''; input.appendChild(img); cell.appendChild(input); menu = doc.body.appendChild(createMenu({ 'set-icon-from-local-file': _('ローカルファイルからアイコンを設定'), 'set-icon-from-url': _('Webページ、または画像ファイルのURLからアイコンを設定'), 'set-icon-from-clipboard': _('クリップボードのURL、または画像データからアイコンを設定'), 'restore-default-icon': _('元のアイコンに戻す'), }, 'icon-menu')); input.setAttribute('menu', menu.id); // 検索エンジン名 headRow.appendChild(doc.createElement('th')).textContent = _('検索エンジン名'); cell = row.appendChild(doc.createElement('th')); input = doc.createElement('input'); input.name = 'name'; cell.appendChild(input); // URL headRow.appendChild(doc.createElement('th')).textContent = _('URL・POSTパラメータ'); cell = row.insertCell(); let urlWrapper = doc.createElement('div'); input = doc.createElement('input'); input.name = 'url'; input.type = 'url'; urlWrapper.appendChild(input); // POSTパラメータ開閉ボタン let params = doc.createElement('button'); params.draggable = true; params.type = 'button'; params.name = 'params'; params.textContent = _('POSTパラメータの設定'); urlWrapper.appendChild(params); cell.appendChild(urlWrapper); // POSTパラメータ let paramsTable = doc.createElement('table'); let paramsTbody = paramsTable.createTBody(); let paramsTemplate = doc.createElement('template'); let paramsRow = paramsTemplate.content.appendChild(doc.createElement('tr')); input = paramsRow.insertCell().appendChild(doc.createElement('input')); input.name = 'post-param-name'; input.placeholder = _('名前'); input = paramsRow.insertCell().appendChild(doc.createElement('input')); input.name = 'post-param-value'; input.placeholder = _('値'); // 行を削除するボタン let cellContainingDeleteRowButton = paramsRow.insertCell(); button = doc.createElement('button'); button.name = 'delete'; button.type = 'button'; img = new Image(); button.title = img.alt = _('行を削除'); img.src = ''; button.appendChild(img); cellContainingDeleteRowButton.appendChild(button); paramsRow.appendChild(cellContainingDeleteRowButton); // 行を追加するボタン let tfootContainingAddRowButton = paramsTable.createTFoot(); button = doc.createElement('button'); button.name = 'add-row'; button.type = 'button'; button.textContent = _('行を追加'); let paramsCell = tfootContainingAddRowButton.insertRow().insertCell(); paramsCell.colSpan = paramsRow.cells.length; paramsCell.appendChild(button); paramsTbody.appendChild(paramsTemplate); cell.appendChild(paramsTable); // メソッド headRow.appendChild(doc.createElement('th')).textContent = _('メソッド'); cell = row.insertCell(); cell.draggable = true; select = doc.createElement('select'); select.name = 'method'; select.add(new Option(_('GET'), 'GET')); select.add(new Option(_('POST'), 'POST')); cell.appendChild(select); // データの種類 headRow.appendChild(doc.createElement('th')).textContent = _('データの種類'); cell = row.insertCell(); cell.draggable = true; select = doc.createElement('select'); select.name = 'accept'; select.add(new Option(_('文字列'), 'text/plain'), true); select.add(new Option(_('画像'), 'image/*')); select.add(new Option(_('音声'), 'audio/*')); cell.appendChild(select); // 文字符号化方式 select = doc.createElement('select'); headRow.appendChild(doc.createElement('th')).textContent = _('文字符号化方式'); cell = row.insertCell(); cell.draggable = true; select.hidden = false; select.name = 'encoding'; let charsetData = CharsetMenu.getData(); for (let encoding of charsetData.pinnedCharsets.concat(charsetData.otherCharsets)) { select.add(new Option(encoding.label, encoding.value, encoding.value === StringUtils.THE_ENCODING)); } cell.appendChild(select); // 行を削除するボタン headRow.appendChild(doc.createElement('th')); row.appendChild(cellContainingDeleteRowButton.cloneNode(true)).draggable = true; template.content.appendChild(row); tbody.appendChild(template); // 行のコンテキストメニュー row.setAttribute('contextmenu', doc.body.appendChild(createMenu({ 'add-row-above': _('上に新しい行を挿入'), 'add-row-below': _('下に新しい行を挿入'), }, 'row-contextmenu')).id); // 行追加ボタン let tfoot = tfootContainingAddRowButton.cloneNode(true); tfoot.getElementsByTagName('td')[0].colSpan = paramsRow.cells.length; table.insertBefore(tfoot, table.tBodies[0]); form.appendChild(table); div = doc.createElement('div'); div.id = 'submit-buttons'; // アイコン一括取得ボタン button = doc.createElement('button'); button.name = 'get-icons'; button.type = 'button'; button.textContent = _('アイコンを一括取得'); button.title = _('アイコン未取得の検索エンジンについて、URLを基にアイコンを取得します。アイコンボタンのポップアップメニューの「元のアイコンに戻す」から、個別に取得することもできます。'); div.appendChild(button); // キャンセルボタン button = doc.createElement('button'); button.name = 'cancel'; button.type = 'button'; button.textContent = _('キャンセル'); div.appendChild(button); // OKボタン button = doc.createElement('button'); button.name = 'ok'; button.textContent = _('OK'); div.appendChild(button); form.appendChild(div); // ブラウザの検索エンジンの追加 div = doc.createElement('div'); div.textContent = _('検索窓のエンジンの追加') + ':'; let addingBrowserEngine = doc.createElement('select'); addingBrowserEngine.name = 'add-browser-engine'; addingBrowserEngine.add(new Option(_('選択してください', '', true, true))); div.appendChild(addingBrowserEngine); form.appendChild(div); // 検索結果を開く場所 label = doc.createElement('label'); label.textContent = _('検索結果を開く場所') + ':'; select = doc.createElement('select'); select.name = 'where'; select.add(new Option(_('現在のタブ。Ctrl、Shiftキーを押していれば、それぞれ新しいタブ、ウィンドウ'), 'current')); select.add(new Option(_('新しいタブ'), 'tab')); select.add(new Option(_('新しいウィンドウ'), 'window')); select.value = Preferences.get(SettingsUtils.ROOT_BRANCH_NAME + 'where', 'tab'); label.appendChild(select); form.appendChild(label); // 検索エンジンの自動追加 label = doc.createElement('label'); let automaticallyReflect = doc.createElement('input'); automaticallyReflect.name = 'automatically-reflect'; automaticallyReflect.type = 'checkbox'; automaticallyReflect.checked = Preferences.get(SettingsUtils.ROOT_BRANCH_NAME + 'automaticallyReflect', true); label.appendChild(automaticallyReflect); label.appendChild(new Text(_('検索窓に新しい検索エンジンが追加されたとき、自動的にドロップゾーンとしても追加する。'))); form.appendChild(label); doc.body.appendChild(form); // 説明 section = doc.createElement('section'); let manual = doc.createElement('ul'); manual.appendChild(doc.createElement('li')).textContent = _('行をドラッグ & ドロップで、順番を変更できます。'); manual.appendChild(doc.createElement('li')).textContent = _('アイコンボタンのポップアップメニューから、アイコンを変更できます。検索窓のエンジンのアイコンは変更できません。'); section.appendChild(manual); doc.body.appendChild(section); // ショートカットの説明 section = doc.createElement('section'); section.id = 'shortcut-keys'; section.appendChild(doc.createElement('h1')).textContent = _('テキスト入力欄のキーボードショートカット'); let shortcuts = doc.createElement('ul'); li = doc.createElement('li'); kbd = doc.createElement('kbd'); key = doc.createElement('kbd'); key.textContent = 'Shift'; key.classList.add('shift'); kbd.appendChild(key); kbd.appendChild(new Text('+')); key = doc.createElement('kbd'); key.textContent = 'Enter'; key.classList.add('enter'); kbd.appendChild(key); li.appendChild(kbd); li.appendChild(new Text(_('上に新しい行を挿入します。'))); shortcuts.appendChild(li); li = doc.createElement('li'); kbd = doc.createElement('kbd'); key = doc.createElement('kbd'); key.textContent = 'Control'; key.classList.add('control'); kbd.appendChild(key); kbd.appendChild(new Text('+')); key = doc.createElement('kbd'); key.textContent = 'Enter'; key.classList.add('enter'); kbd.appendChild(key); li.appendChild(kbd); li.appendChild(new Text(_('または'))); kbd = doc.createElement('kbd'); key = doc.createElement('kbd'); key.textContent = 'Alt'; key.classList.add('alt'); kbd.appendChild(key); kbd.appendChild(new Text('+')); key = doc.createElement('kbd'); key.textContent = 'Enter'; key.classList.add('enter'); kbd.appendChild(key); li.appendChild(kbd); li.appendChild(new Text(_('下に新しい行を挿入します。'))); shortcuts.appendChild(li); li = doc.createElement('li'); kbd = doc.createElement('kbd'); key = doc.createElement('kbd'); key.textContent = 'Control'; key.classList.add('control'); kbd.appendChild(key); kbd.appendChild(new Text('+')); kbd.appendChild(doc.createElement('kbd')).textContent = '↑'; li.appendChild(kbd); li.appendChild(new Text(_('または'))); kbd = doc.createElement('kbd'); key = doc.createElement('kbd'); key.textContent = 'Alt'; key.classList.add('alt'); kbd.appendChild(key); kbd.appendChild(new Text('+')); kbd.appendChild(doc.createElement('kbd')).textContent = '↑'; li.appendChild(kbd); li.appendChild(new Text(_('上の行に移動します。'))); shortcuts.appendChild(li); li = doc.createElement('li'); kbd = doc.createElement('kbd'); key = doc.createElement('kbd'); key.textContent = 'Control'; key.classList.add('control'); kbd.appendChild(key); kbd.appendChild(new Text('+')); kbd.appendChild(doc.createElement('kbd')).textContent = '↓'; li.appendChild(kbd); li.appendChild(new Text(_('または'))); kbd = doc.createElement('kbd'); key = doc.createElement('kbd'); key.textContent = 'Alt'; key.classList.add('alt'); kbd.appendChild(key); kbd.appendChild(new Text('+')); kbd.appendChild(doc.createElement('kbd')).textContent = '↓'; li.appendChild(kbd); li.appendChild(new Text(_('下の行に移動します。'))); shortcuts.appendChild(li); section.appendChild(shortcuts); doc.body.appendChild(section); // インポートとエクスポート section = doc.createElement('section'); section.appendChild(doc.createElement('h1')).textContent = _('インポートとエクスポート'); dl = doc.createElement('dl'); button = doc.createElement('button'); button.name = 'export'; button.type = 'button'; button.textContent = _('エクスポート'); dl.appendChild(doc.createElement('dt')).appendChild(button); dl.appendChild(doc.createElement('dd')).textContent = _('現在の設定をファイルへエクスポートします。保存していない設定は反映されません。'); button = doc.createElement('button'); button.name = 'import'; button.type = 'button'; button.textContent = _('インポート'); dl.appendChild(doc.createElement('dt')).appendChild(button); dl.appendChild(doc.createElement('dd')).textContent = _('現在の設定をすべて削除し、XMLファイルから設定をインポートします。ブラウザの検索エンジンサービスに同名の検索エンジンが存在する場合は、そちらを優先します。'); button = doc.createElement('button'); button.name = 'additional-import'; button.type = 'button'; button.textContent = _('追加インポート'); dl.appendChild(doc.createElement('dt')).appendChild(button); dl.appendChild(doc.createElement('dd')).textContent = _('XMLファイルから検索エンジンを追加します。同名の検索エンジンがすでに存在する場合は上書きします。'); button = doc.createElement('button'); button.name = 'import-from-text'; button.type = 'button'; button.textContent = _('JSON文字列から追加インポート'); dl.appendChild(doc.createElement('dt')).appendChild(button); dl.appendChild(doc.createElement('dd')).textContent = _('本スクリプトのバージョン1でエクスポートしたJSON文字列から、検索エンジンを追加します。'); section.appendChild(dl); doc.body.appendChild(section); // 初期化やアンインストールについて section = doc.createElement('section'); section.appendChild(doc.createElement('h1')).textContent = _('その他'); dl = doc.createElement('dl'); button = doc.createElement('button'); button.name = 'initialize'; button.type = 'button'; button.textContent = _('設定を初期化'); dl.appendChild(doc.createElement('dt')).appendChild(button); dl.appendChild(doc.createElement('dd')).textContent = _('すべての設定を削除し、初回起動時の状態に戻します。'); button = doc.createElement('button'); button.name = 'uninstall'; button.type = 'button'; button.textContent = _('すべての設定を削除'); dl.appendChild(doc.createElement('dt')).appendChild(button); dl.appendChild(doc.createElement('dd')).textContent = _('すべての設定を削除し、スクリプトを停止します。'); section.appendChild(dl); doc.body.appendChild(section); /** * メニューを作成する * @param {Object} commands - キーがmenuitem要素のid属性値、値がlabel属性値の連想配列 * @param {string} id - menu要素のid属性値 * @returns {HTMLMenuElement} menu要素 */ function createMenu(commands, id) { let menu = document.createElementNS(DOMUtils.HTML_NS, 'menu'); menu.id = id; for (let id in commands) { let menuitem = document.createElementNS(DOMUtils.HTML_NS, 'menuitem'); menuitem.id = id; menuitem.label = commands[id]; menu.appendChild(menuitem); } if (menu.type !== 'popup') { // Bug 897102 – Update <menu> to spec <https://bugzilla.mozilla.org/show_bug.cgi?id=897102> menu.type = 'context'; } return menu; } }, }; /** * フォームデータ集合の各エントリ。 * @typedef FormDataEntry * @type {Array} * @property {string} 0 - 名前。 * @property {(string|nsIMIMEInputStream)} 1 - 値。 * @see [Interface FormData - XMLHttpRequest Standard]{@link http://xhr.spec.whatwg.org/#interface-formdata} */ /** * 文字列操作。 */ let StringUtils = { /** * [Encoding Standard]{@link http://encoding.spec.whatwg.org/}が要求する標準の文字符号化方式。 * @constant {string} */ THE_ENCODING: 'UTF-8', /** * 境界文字列の前半に用いるハイフンマイナスの数。 * @type {number} */ BOUNDARY_HYPHEN_LENGTH: 25, /** * 境界文字列の後半に用いる乱数値の文字数。 * @type {number} */ BOUNDARY_RANDOM_STRING_LENGTH: 15, /** * フォームデータ集合を multipart/form-data として{@link nsIInputStream}に変換する。 * @param {FormDataEntry[]} formDataSet * @param {Function} callback - 第1引数に戻り値としての{@link nsIMIMEInputStream}を含むコールバック関数。 * @param {string} [encoding=StringUtils.THE_ENCODING] */ encodeMultipartFormData: function (formDataSet, callback, encoding = this.THE_ENCODING) { // 境界文字列を生成 let boundary = '-'.repeat(this.BOUNDARY_HYPHEN_LENGTH) + (Math.random() * Math.pow(10, this.BOUNDARY_RANDOM_STRING_LENGTH)).toFixed(); // 要求本体の生成 let multipartBody = new MultiplexInputStream(); for (let i = 0, l = formDataSet.length; i < l; i++) { let [name, value] = formDataSet[i]; multipartBody.appendStream(new StringInputStream((i > 0 ? '\r\n' : '') + '--' + boundary + '\r\n', -1)); let isMIMEInputStream = value instanceof Ci.nsIMIMEInputStream; let bodyPart = isMIMEInputStream ? value : new MIMEInputStream(); bodyPart.addHeader('content-disposition', 'form-data; name=' + StringUtils.quote(StringUtils.convertEncoding(name, encoding)) + (isMIMEInputStream ? '; filename=' + StringUtils.quote(StringUtils.convertEncoding('blob', encoding)) : '')); if (isMIMEInputStream) { multipartBody.appendStream(bodyPart); } else { bodyPart.setData(StringUtils.convertToInputStream(value, encoding)); multipartBody.appendStream(bodyPart); } } multipartBody.appendStream(new StringInputStream('\r\n--' + boundary + '--\r\n', -1)); // 要求ヘッダの設定 let postData = new MIMEInputStream(); postData.addHeader('content-type', 'multipart/form-data; boundary=' + boundary); postData.addContentLength = true; // 要求本体の設定 postData.setData(multipartBody); callback(postData); }, /** * 文字列を指定した符号化方式の{@link nsIInputStream}として返す。 * @param {string} str * @param {string} [encoding=StringUtils.THE_ENCODING] * @returns {nsIInputStream} */ convertToInputStream: function (str, encoding = this.THE_ENCODING) { try { ScriptableUnicodeConverter.charset = encoding; } catch (e) { if (e.result === Cr.NS_ERROR_UCONV_NOCONV) { ScriptableUnicodeConverter.charset = this.THE_ENCODING; } else { throw e; } } return ScriptableUnicodeConverter.convertToInputStream(str); }, /** * 文字列を指定した符号化方式のバイナリ文字列に変換する。 * @param {string} str * @param {string} [encoding=StringUtils.THE_ENCODING] * @returns {string} */ convertEncoding: function (str, encoding = this.THE_ENCODING) { let stream = this.convertToInputStream(str, encoding); return NetUtil.readInputStreamToString(stream, stream.available()); }, /** * 文字列をquoted-string形式に。(", \, CR, LF にバックスラッシュを前置) * @param {string} str * @returns {string} */ quote: function (str) { return '"' + str.replace(/["\\\r\n]/g, '\\$&') + '"'; }, }; /** * アンインストール時に実行する処理。 */ let UninstallObserver = { /** * 通知の種類。 * @constant {string} */ TYPE: 'uninstall', /** * スクリプトが停止していれば真。 * @type {booelan} */ uninstalled: false, /** * オブザーバを登録。 */ init: function () { ObserverUtils.register(this.TYPE, this); }, /** * オブザーバ。 */ observe: function () { this.uninstalled = true; ObserverUtils.stop(); document.getElementById(DragAndDropZonesPlus.ID + '-menuitem').remove(); DropzoneUtils.remove(); DropzoneUtils.removeEventListeners(); }, /** * 通知。 */ notify: function () { ObserverUtils.notify(this.TYPE); }, }; /** * クリップボードに関する操作。 */ let ClipboardUtils = { /** * クリップボードからテキスト情報を取得する。 * @param {number} whichClipboard {@link Services.clipboard.kSelectionClipboard}か{@link Services.clipboard.kGlobalClipboard} * @returns {?string} */ getText: function (whichClipboard) { if (Services.clipboard.hasDataMatchingFlavors(['text/unicode'], 1, whichClipboard)) { // テキストデータが保持されていれば let transferable = new Transferable('text/unicode'), data = {}; Services.clipboard.getData(transferable, whichClipboard); transferable.getTransferData('text/unicode', data, {}); return data.value.QueryInterface(Ci.nsISupportsString).data; } else { return null; } }, }; /** * DOM関連のメソッド。 */ let DOMUtils = { /** * HTML名前空間。 * @constant {string} */ HTML_NS: 'http://www.w3.org/1999/xhtml', /** * 属性値を{@link DOMTokenList}として取得する。 * @param {Element} element - 要素。 * @param {string} attributeName - 属性値名。 * @returns {DOMTokenList} * @see {@link https://dom.spec.whatwg.org/#interface-domtokenlist 7.1. Interface DOMTokenList | DOM Standard} */ getAttributeAsDOMTokenList: function (element, attributeName) { let tokenList = document.createElementNS(this.HTML_NS, 'div').classList; tokenList.value = element.getAttribute(attributeName) || ''; return tokenList; }, /** * ノードに対応するfigcaption要素を取得する。 * @param {Node} node * @returns {?HTMLElement} */ getFigcaption: function (node) { let figcaption = null; let parent = node.parentElement; if (parent && parent.localName === 'figure') { let first = parent.firstElementChild; if (first) { if (first.localName === 'figcaption') { figcaption = first; } else { let last = parent.lastElementChild; if (last && last.localName === 'figcaption') { figcaption = last; } } } } return figcaption; }, /** * 指定した局所名を持つ直近の親を返す。 * @param {Node} childNode * @param {string} localName * @returns {?Element} */ getParentElementByTagName: (function () { let _localName; let treeWalkers = new WeakMap(); return function (childNode, localName) { if (childNode.localName === localName) { return childNode; } else { let doc = childNode.ownerDocument; let treeWalker = treeWalkers.get(doc); if (!treeWalker) { treeWalker = doc.createTreeWalker(doc, NodeFilter.SHOW_ELEMENT, function (node) { return node.localName === _localName ? NodeFilter.FILTER_ACCEPT : NodeFilter.FILTER_SKIP; }); treeWalkers.set(doc, treeWalker); } _localName = localName; treeWalker.currentNode = childNode; return treeWalker.parentNode(); } }; })(), /** * 指定されたノードの子孫要素に、適宜改行とインデントを挿入し読みやすくする。 * すでに改行やインデントが含まれていることは想定しない。 * @param {Node} root * @returns {Node} */ toPrettyXML: function (root) { let walker = (root.ownerDocument || root).createTreeWalker(root, NodeFilter.SHOW_ELEMENT); let indent = 0; while (true) { if (walker.firstChild()) { // 子要素を操作し、存在すればインデントを増やす indent++; } else { // 子要素が存在しなければ、次の同胞要素が存在する親要素を走査する while (true) { if (walker.nextSibling()) { break; } else if (walker.parentNode()) { indent--; } else { // すべての要素を走査し終えていれば return root; } } } // 要素の前に改行とインデントを挿入する walker.currentNode.parentNode.insertBefore(new Text('\n' + '\t'.repeat(indent)), walker.currentNode); if (!walker.currentNode.nextElementSibling) { // 次の同胞要素が存在しなければ、要素の後ろに改行とインデントを追加する walker.currentNode.parentNode.appendChild(new Text('\n' + '\t'.repeat(indent - 1))); } } }, /** * 選択範囲と指定した座標が重なるか調べる。 * @param {Selection} selection * @param {number} x * @param {number} y * @returns {boolean} */ isSuperposedCoordinateOnSelection: function (selection, x, y) { for (let i = 0, l = selection.rangeCount; i < l; i++) { if (this.isSuperposedCoordinateOnRange(selection.getRangeAt(i), x, y)) { return true; } } return false; }, /** * rangeと指定した座標が重なるか調べる。 * @param {Range} range * @param {number} x * @param {number} y * @returns {boolean} */ isSuperposedCoordinateOnRange: function (range, x, y) { return Array.prototype.some.call(range.getClientRects(), rect => { return this.isSuperposedCoordinateOnRect(rect, x, y); }); }, /** * 長方形と指定した座標が重なるか調べる。 * @param {DOMRect} rect * @param {number} x * @param {number} y * @returns {boolean} */ isSuperposedCoordinateOnRect: function (rect, x, y) { return x >= rect.left && x <= rect.right && y >= rect.top && y <= rect.bottom; }, }; /** * オブザーバサービスを利用して、各ウィンドウと通知を送受信する。 * @version 2014-05-10 */ let ObserverUtils = { /** * オブザーバを追加する。 * @param {string} id - 当スクリプトに関する通知のID。 */ init: function (id) { this.id = id; Services.obs.addObserver(this, this.id, false); window.addEventListener('unload', () => this.stop()); }, /** * オブザーバを削除する。 */ stop: function () { Services.obs.removeObserver(this, this.id); }, /** * 通知を受け取る関数を登録する。同じtypeの場合は上書きされる。 * @param {string} type * @param {Object} observer - observeメソッドを持つオブジェクト。 */ register: function (type, observer) { this.observers[type] = observer; }, /** * 通知。 * @param {string} type * @param {*} [data] */ notify: function (type, data = null) { Services.obs.notifyObservers(data, this.id, type); }, /** * オブザーバのリスト。 * @type {Object} * @access protected */ observers: {}, /** * 当スクリプトに関する通知のID。 * @type {string} * @access protected */ id: null, /** * 通知を受け取るメソッド。 * @param {*} subject * @param {string} topic * @param {string} data * @access protected */ observe: function (subject, topic, data) { if (data in this.observers) { // 対応するオブザーバが存在すれば this.observers[data].observe(subject); } }, }; /** * ポップアップ通知を表示する。 * @param {string} message - 表示するメッセージ。 * @param {XULElement} tab - メッセージを表示するタブ。 * @param {string} [type=information] - メッセージの前に表示するアイコンの種類。"information"、"warning"、"error"、"question" のいずれか。 * @version 2016-06-09-drag-and-drop-zones-plus */ function showPopupNotification(message, tab, type = 'information') { let win = tab.ownerDocument.defaultView; if (win.closed) { // 指定されたタブを含むウィンドウが既に閉じていれば、別ウィンドウの最前面のタブを取得 tab = Services.wm.getMostRecentWindow('navigator:browser').gBrowser.selectedTab; } else if (!tab.ownerDocument.contains(tab)) { // 指定されたタブが既に閉じていれば、最前面のタブを取得 tab = win.gBrowser.selectedTab; } win.PopupNotifications.show(gBrowser.getBrowserForTab(tab), DragAndDropZonesPlus.ID, message, null, null, null, { persistWhileVisible: true, removeOnDismissal: true, popupIconURL: 'chrome://global/skin/icons/' + type + '-' + (type === 'information' ? '32' : '64') + '.png', }); } DragAndDropZonesPlus.main(); })();