您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
AutoPagerizeの操作系を拡張します。
当前为
// ==UserScript== // @name AutoPagerize_Console // @name:en AutoPagerize_Console // @description AutoPagerizeの操作系を拡張します。 // @description:en Expansion Autopagerize operation. // @namespace phodra // @include * // @exclude https://www.youtube.com/* // @version 5.02 // @noframes // @grant GM_getValue // @grant GM_setValue // @grant GM.getValue // @grant GM.setValue // ==/UserScript== (()=>{ // Greasemonkey 4.0未満バージョン対応 if( this.GM == null ){ this.GM = {}; this.GM.getValue = async (name, defaultValue) => { return GM_getValue(name, defaultValue); }; this.GM.setValue = async (name, value) => { return GM_setValue(name, value); }; } // 拡張 Array.prototype.first = function(){ return this[0]; }; Array.prototype.last = function(){ return this[this.length-1]; }; Array.prototype.round = function(i){ return this[i<0? 0: i>=this.length? this.length-1: i]; }; Node.prototype.prependChild = function(elm){ this.insertBefore(elm,this.firstChild); }; Element.prototype.css = function(arg1, arg2){ if( arg2 == null){ for( let key in arg1 ){ this.style[key] = arg1[key]; } }else{ this.style[arg1] = arg2; } }; Element.prototype.attr = function(arg1, arg2){ if( arg2 == null){ for( let key in arg1 ){ this.setAttribute(key, arg1[key]); } }else{ this.setAttribute(arg1, arg2); } }; // 任意のイベントを発火 const FireEvent = ename => { const e = document.createEvent('Event'); e.initEvent( ename, true, false); return document.dispatchEvent(e); }; // 画像をひとまとめにしておく const RES = { 'config': "", 'scrAll': "", 'scrPage': "", 'disabled': "", 'enabled': "", }; // スタイル const $style = document.createElement("style"); $style.setAttribute("type", "text/css"); $style.textContent = ` #apc-panel *, #apc-config * { color: #eee; border-color: #666; } .apc-box, #apc-pageList, #apc-config { background-color: #1c1c1c !important; } .apc-button { filter: brightness(100%) contrast(150%); } #apc-enabler { filter: grayscale(100%); } #apc-panel[valid] #apc-enabler { filter: grayscale(60%); } /* コンソールパネル */ #apc-panel { opacity: 0.6; background-color: transparent !important; position: fixed; z-index: 9999999990; right: 0; /*top: 48px;*/ margin: 0; display: inline-flex; flex-direction: column; align-items: flex-end; } #apc-panel[display_rule=valid]:not([valid]) { visibility: hidden; } #apc-panel *, #apc-config * { font-family: arial, sans-serif; font-size: 14px; font-weight: normal; letter-spacing: normal; vertical-align: baseline; line-height: normal; } .apc-box { padding: 1px; box-sizing: content-box; min-width: 18px; /*position: relative;*/ pointer-events: auto; } #apc-panel[non_hover='transparent']:not(:hover):not([config]) .apc-box { background-color: transparent !important; } #apc-panel[non_hover='stealth']:not(:hover):not([config]) .apc-box { visibility: hidden; } /* ボタンのスタイル */ .apc-button { border: solid 1px; box-sizing: border-box; cursor: pointer; margin: 1px; padding: 0; position: relative; opacity: 0.5; width: 16px; } .apc-button:hover { opacity: 0.8; } .apc-button:active { opacity: 0.3; } /* ボタンの大きさ */ #apc-optionBox .apc-button { height: 16px; } #apc-scrollerBox .apc-button { height: 32px; } /* オプションボックス */ #apc-optionBox { display: flex; z-index: 9999999991; } #apc-enabler[state="enable"] { background-image: url(${RES.enabled}); } #apc-enabler[state="disable"] { background-image: url(${RES.disabled}); } #apc-setting { background-image: url(${RES.config}); } /* optionBox配置形状: */ #apc-panel[valid] #apc-optionBox[option_dir_valid='Left'], #apc-panel:not([valid]) #apc-optionBox[option_dir_invalid='Left'] { flex-direction: row-reverse; } #apc-panel[valid] #apc-optionBox[option_dir_valid='Right'], #apc-panel:not([valid]) #apc-optionBox[option_dir_invalid='Right'] { flex-direction: row; } #apc-panel[valid] #apc-optionBox[option_dir_valid^='Up'], #apc-panel:not([valid]) #apc-optionBox[option_dir_invalid^='Up'] { flex-direction: column-reverse; } #apc-panel[valid] #apc-optionBox[option_dir_valid^='Down'], #apc-panel:not([valid]) #apc-optionBox[option_dir_invalid^='Down'] { flex-direction: column; } #apc-panel[valid] #apc-optionBox[option_dir_valid='Up_protrude'], #apc-panel:not([valid]) #apc-optionBox[option_dir_invalid='Up_protrude'] { margin-top: -18px; } #apc-panel[valid] #apc-optionBox[option_dir_valid='Down_protrude'], #apc-panel:not([valid]) #apc-optionBox[option_dir_invalid='Down_protrude'] { margin-bottom: -18px; } /* スクローラーボックス */ #apc-scrollerBox { z-index: 9999999992; } #apc-scrollTop, #apc-scrollBottom { background-image: url(${RES.scrAll}); } #apc-scrollPrev, #apc-scrollNext { background-image: url(${RES.scrPage}); } /* scrollerBox配置形状:スリム */ #apc-scrollerBox[scroller_form='Slim'] { display: flex; flex-flow: column wrap-reverse; } #apc-scrollerBox[scroller_form='Slim'] #apc-scrollTop { order: 1; } #apc-scrollerBox[scroller_form='Slim'] #apc-scrollPrev { order: 2; } #apc-scrollerBox[scroller_form='Slim'] #apc-scrollNext { order: 3; transform: scale(1, -1); } #apc-scrollerBox[scroller_form='Slim'] #apc-scrollBottom, #apc-panel:not([valid]) #apc-scrollBottom { order: 4; transform: scale(1, -1); } /* scrollerBox配置形状:スクエア */ #apc-panel[valid] #apc-scrollerBox[scroller_form='Square'] { display: grid; } #apc-panel[valid] #apc-scrollerBox[scroller_form='Square'] #apc-scrollTop { grid-row: 1/2; grid-column: 2/3; transform: scale(-1, 1); } #apc-panel[valid] #apc-scrollerBox[scroller_form='Square'] #apc-scrollBottom { grid-row: 2/3; grid-column: 2/3; transform: scale(-1, -1); } #apc-panel[valid] #apc-scrollerBox[scroller_form='Square'] #apc-scrollNext { grid-row: 2/3; grid-column: 1/2; transform: scale(1, -1); } #apc-panel[valid] #apc-scrollerBox[scroller_form='Square'] #apc-scrollPrev { grid-row: 1/2; grid-column: 1/2; } /* ページボックス */ #apc-pageIndexBox { cursor: pointer; z-index: 9999999993; min-height: 17px; position: relative; } /* ページ表示 */ #apc-sequencer { margin: 0px 1px; padding: 0px; text-align: center; } /* pageIndexBox配置形状:縦置き */ #apc-pageIndexBox[page_index_form$='Pile'] #apc-sequencer { display: flex; flex-direction: column; } #apc-pageIndexBox[page_index_form$='Pile'] span:first-child{ border-bottom: solid 1px #eee !important; } /* pageIndexBox配置形状:横置き */ #apc-pageIndexBox[page_index_form='Strip'], #apc-pageIndexBox[page_index_form='Growed Pile']{ box-sizing: border-box; width: 100%; } #apc-pageIndexBox[page_index_form$='Strip'] span:first-child::after{ content: " / "; } /* ページ一覧 */ #apc-pageList { position: absolute; min-width: 38px; margin: 0; padding: 0px; list-style-type: none; display: none; flex-flow: column wrap-reverse; max-height: 50vh; } #apc-pageIndexBox:hover #apc-pageList { display: flex; } #apc-pageIndexBox[page_list_expand^="Left"] #apc-pageList { right: 100%; } #apc-pageIndexBox[page_list_expand^="Right"] #apc-pageList { left: 100%; flex-wrap: wrap; } #apc-pageIndexBox[page_list_expand$="upper"] #apc-pageList { bottom: 0; } #apc-pageIndexBox[page_list_expand$="lower"] #apc-pageList { top: 0; } #apc-pageIndexBox[page_list_expand="Upper"] #apc-pageList { right: 0; bottom: 100%; } #apc-pageIndexBox[page_list_expand="Lower"] #apc-pageList { right: 0; top: 100%; } .apc-pageListItem { border-style: outset; border-width: 1px; box-sizing: border-box; cursor: pointer; margin: 1px; padding: 0px; text-align: center; } #apc-panel:not([valid]) #apc-scrollPrev, #apc-panel:not([valid]) #apc-scrollNext, #apc-panel:not([valid]) #apc-pageIndexBox { display: none; } #apc-panel[config]+#apc-configDialog { display: flex; } /* コンフィグメニューダイアログ */ #apc-configDialog { background-color: transparent !important; position: fixed; z-index: 9999999999; top: 0px; left: 0px; width: 100%; height: 100%; display: none; flex-direction: column; justify-content: center; align-items: center; } #apc-config { opacity: 1.0; border: outset 3px white; display: inline-block; text-align: left; padding: 5px 15px; max-height: 90vh; overflow: scroll; user-select: none; -moz-user-select: none; -webkit-user-select: none; -ms-user-select: none; } #apc-config h4 { display: block; background-color: transparent; padding: 0; padding-bottom: 1px; margin: 5px 0 2px -5px; border: solid 0; border-bottom-width: 1px; font-weight: bold !important; } #apc-config .apc-group { border: solid 1px; /*position: relative;*/ margin: 0; margin-top: 10px; padding: 5px; font-size: 0; } #apc-config .apc-group h5 { /*position: relative;*/ display: inline-block; margin: 0; margin-top: -1em; padding: 0 3px; background-color: #1c1c1c; font-weight: bold !important; } #apc-config .apc-mItem { display: block; margin: 5px 2px 0 2px; } #apc-config label { display: block; margin: 2px; cursor: pointer; } #apc-config input, #apc-config select { /*color: white;*/ background-image: none; background-color: #000; border: 2px inset #666; margin: auto 2px; cursor: pointer; border-radius: initial; padding: initial; } #apc-config option { color: black; background-color: white; } #apc-config input { width: auto; height: auto; padding: 0; cursor: pointer; } #apc-config select, #apc-config input[type='number'] { box-sizing: content-box; height: 19px; } #apc-config input[type='number'] { max-width: 60px; -webkit-appearance: ; } #apc-config select { max-width: 80px; } #apc-config input[type='checkbox'] { vertical-align: middle; } #apc-config #apc-mButtons { display: flex; justify-content: stretch; margin: 10px 0 5px 0; } #apc-config .apc-config_button { border: outset 2px; margin: 0 5px; width: 100%; } #apc-config .apc-config_button:first-child { margin-left: 0; } #apc-config .apc-config_button:last-child { margin-right: 0; } `; document.head.appendChild($style); // APが有効なページであるのか const apValid = { name: "valid", flag: null, }; /// APメッセージバー(アドオン版)かAPアイコン(スクリプト版)が /// 挿入されると、APが有効なページであるとみなす。 const checkApValid = $elm => { if( apValid.flag == null ){ // AutoPagerize有効なページでのみ行う処理 if( $elm.id == "autopagerize_message_bar" || $elm.id == "autopagerize_icon" ) { apValid.flag = $elm.id; $panel.attr( apValid.name, true); return true; } } return false; }; const apObject = document.getElementById("autopagerize_message_bar") || document.getElementById("autopagerize_icon"); if( apObject != null ){ checkApValid(apObject); }else{ const mo = new MutationObserver( mRecs => { for( let rec of mRecs ){ for( let added of rec.addedNodes ){ if( checkApValid(added) ){ mo.disconnect(); } } } } ); mo.observe( document.body, {'childList': true}); document.addEventListener( 'GM_AutoPagerizeNextPageLoaded', () => mo.disconnect() ); } let scrTop = document.documentElement.scrollTop; let oldPage = null; // ウィンドウのスクロールが発生した時 window.addEventListener( 'scroll', () => { scrTop = document.documentElement.scrollTop; // 現在のページを更新 const np = getNowPage(); if( np != oldPage ){ $sequencerNow.textContent = np+1; oldPage = np; } } ); // 各ページを管理する配列 let pageBounds = [0]; // 現在のページを取得(境界線上を前ページとみなす場合false) const getNowPage = (notLess=true) => { const breakOver = notLess ? (st, bound) => st >= bound : (st, bound) => st > bound ; let i; for( i = pageBounds.length-1; i >= 0; i-- ){ if( breakOver( scrTop, pageBounds[i]) ) break; } return i; }; const getBottomPos = () => { return Math.max( document.body.clientHeight, document.body.scrollHeight, document.documentElement.scrollHeight, document.documentElement.clientHeight) - window.innerHeight; }; // ページを継ぎ足した時、継ぎ目の位置を記録する const apPageAppended = () => { // セパレーターの絶対位置を取得 const $apSep = document.getElementsByClassName("autopagerize_page_separator"); const len = $apSep.length; let sepPos = $apSep.item(len-1).getBoundingClientRect().top + window.pageYOffset; sepPos = Math.round(sepPos); if( !pageBounds.includes(sepPos) ){ pageBounds.push(sepPos); } $sequencerMax.textContent = len+1; // ページリストアイテムを追加 $pageList.appendChild( $createPageListItem(len+1) ); }; document.addEventListener( 'GM_AutoPagerizeNextPageLoaded', () => apPageAppended() ); let scrTick = null; // 任意の位置にスクロール const smoothScrollTo = targetY => { // 続けてスクロールさせるとキャンセルして新たなスクロール if( scrTick != null ){ clearTimeout(scrTick); scrTick = null; } // 再帰処理部分 const smoothScrollTick = (_targetY) => { // 常に最新の位置をターゲットにしたい場合、引数に関数を渡す // (最下部へのスクロール用。 // なぜかスクロール途中にページの高さが変わる場合がある) let targetY = (typeof(_targetY) == "function" ? _targetY() : _targetY); // 移動量 let moveY = (targetY-scrTop)/10; // 終了条件 let endOver; // 下方向の移動量調整と終了条件 if( moveY>0 ){ moveY = Math.ceil(moveY); //切り上げ endOver = scrTop+moveY>=targetY; //ターゲットを越えていれば } // 上方向の移動量調整と終了条件 else{ moveY = Math.floor(moveY); //切り下げ endOver = scrTop+moveY<=targetY; //ターゲットを下回っていれば } // 条件を満たしていれば終了 if( endOver ){ window.scrollTo( 0, targetY); clearTimeout(scrTick); scrTick = null; }else{ window.scrollBy( 0, moveY); scrTick = setTimeout( () => smoothScrollTick(targetY), 10 ); } }; smoothScrollTick(targetY); }; // コントロール配置 /// パネル(最親) const $panel = document.createElement("div"); $panel.id = "apc-panel"; document.body.appendChild($panel); // ボックスを作成 const $createBox = () => { const $box = document.createElement("div"); $box.className = "apc-box"; return $box; }; // ボタンを作成 const $createButton = (attr) => { const $button = document.createElement("div"); $button.className = "apc-button"; $button.attr(attr); return $button; }; ///panel/ オプションボックス const $optionBox = $createBox(); $optionBox.id = "apc-optionBox"; $panel.appendChild($optionBox); ///panel/optionBox/ オンオフボタン (enable/disable) const $enabler = $createButton( { 'id': "apc-enabler" } ); const apEnable = { name: "enable", state: true, }; const changeToggle = (toggle = null) => { apEnable.state = toggle != null ? toggle : !apEnable.state; $enabler.attr( apEnable.state ? { 'state': "enable", 'alt': "E", 'title': "AutePagerize OFF (Now:Enable)" } : { 'state': "disable", 'alt': "D", 'title': "AutePagerize ON (Now:Disable)" } ); GM.setValue( apEnable.name, apEnable.state); }; ///panel/optionBox// APイベントをキャプチャー document.addEventListener( 'AutoPagerizeToggleRequest', () => changeToggle() ); document.addEventListener( 'AutoPagerizeEnableRequest', () => changeToggle(true) ); document.addEventListener( 'AutoPagerizeDisableRequest', () => changeToggle(false) ); ///panel/optionBox// トグルボタン クリックでリクエストイベント発火 $enabler.addEventListener( 'click', () => FireEvent('AutoPagerizeToggleRequest') ); $optionBox.appendChild($enabler); // コンフィグメニューを開いているか ///panel/optionBox/ 設定ボタン const $setting = $createButton( { 'id': "apc-setting", 'title': "Open Config", 'alt': "c", } ); $optionBox.appendChild($setting); $setting.addEventListener( 'click', () => config.open() ); /// ボタンが見えなくなった時のため、ショートカットで開けるようにする。 /// Alt + Ctrl + p document.addEventListener( 'keydown', e => { if( e.altKey && e.ctrlKey && e.keyCode == 80 ){ config.open(); } } ); const $spacingBox12 = document.createElement("div"); $spacingBox12.className = "apc-spacing"; $spacingBox12.style.order = 2; $panel.appendChild($spacingBox12); ///panel/ スクロールボックス const $scrollerBox = $createBox(); $scrollerBox.id = "apc-scrollerBox"; $panel.appendChild($scrollerBox); ///panel/scr/ 最上部へ移動 const $scrollTop = $createButton( { 'id': "apc-scrollTop", 'alt': "↑", 'title': "Move to Top", } ); $scrollerBox.appendChild($scrollTop); ///panel/scr/ 最下部へ移動 const $scrollBottom = $createButton( { 'id': "apc-scrollBottom", 'alt': "↓", 'title': "Move to Bottom", } ); $scrollerBox.appendChild($scrollBottom); ///panel/scr/ 前のページ const $scrollPrev = $createButton( { 'id': "apc-scrollPrev", 'alt': "△", 'title': "Move to Previous", } ); $scrollerBox.appendChild($scrollPrev); ///panel/scr/ 次のページ const $scrollNext = $createButton( { 'id': "apc-scrollNext", 'alt': "▽", 'title': "Move to Next", } ); $scrollerBox.appendChild($scrollNext); ///panel/scr/ 最上部へ移動 $scrollTop.addEventListener( 'click', () => smoothScrollTo(0) ); ///panel/scr/ 最下部へ移動 $scrollBottom.addEventListener( 'click', () => smoothScrollTo(getBottomPos) ); ///panel/scr/ 前のページ $scrollPrev.addEventListener( 'click', () => { smoothScrollTo( pageBounds.round( getNowPage(false) ) ); } ); ///panel/scr/ 次のページ $scrollNext.addEventListener( 'click', () => { const targetPage = getNowPage()+1; smoothScrollTo( targetPage < pageBounds.length ? pageBounds.round(targetPage) : getBottomPos ); } ); const $spacingBox23 = document.createElement("div"); $spacingBox23.className = "apc-spacing" $spacingBox23.style.order = 4; $panel.appendChild($spacingBox23); ///panel/ ページボックス const $pageIndexBox = $createBox(); $pageIndexBox.id = "apc-pageIndexBox"; ///panel/pageIndexBox/ ページ数表示 const $sequencer = document.createElement("div"); $sequencer.id = "apc-sequencer"; const $sequencerNow = document.createElement("span"); const $sequencerMax = document.createElement("span"); $sequencerNow.textContent = $sequencerMax.textContent = "1"; $sequencer.appendChild($sequencerNow); $sequencer.appendChild($sequencerMax); $pageIndexBox.appendChild($sequencer); $panel.appendChild($pageIndexBox); ///panel/pageIndexBox/ ページリスト const $pageList = document.createElement("ol"); $pageList.id = "apc-pageList"; $pageIndexBox.appendChild($pageList); // 新しいページリストアイテムにイベントを追加 const $createPageListItem = num => { const $elm = document.createElement("li"); $elm.className = "apc-pageListItem"; $elm.textContent = num; // クリックでそのページにスクロール $elm.addEventListener( 'click', function(){ const targetNum = this.textContent-1; smoothScrollTo( pageBounds.round(targetNum) ); } ); // // ダブルクリックでページ移動 // $elm.addEventListener( // 'dblclick', function(){ // const num = this.textContent-2; // if( num >= 0 ) // document.getElementsByClassName("autopagerize_link")[num].href; // } // ); return $elm; }; $pageList.appendChild( $createPageListItem(1) ); /// コンフィグメニューを作成 // コンフィグコントロール管理クラス // 基底クラス class ConfigControl { constructor(name, options){ this.name = name; this.defaultValue = options.defaultValue || 0; this.onchange_ = options.onchange || null; } get state(){ return this.value; } set state(val){ this.value = val; this.onchange(); } onchange(){ if( typeof(this.onchange_) == "function" ) this.onchange_(); } getInitial(){ return this.defaultValue || 0; } setDefault(){ this.state = this.defaultValue; } restoreState(params){ const val = params[this.name]; if( val == null ){ this.setDefault(); }else{ this.state = val; } } /* 仮想関数 * 継承先のクラスでこれらをオーバーライドする * オーバーライドされていなければバグなのでwarn */ /* virtual */get value(){ console.warn("virtual get value", this.name); return null; } /* virtual */set value(val){ console.warn("virtual set value", this.name); } /* virtual */appendTo($to){ console.warn("virtual appendTo", this.name); } } // NumericUpDown管理クラス class NumericSet extends ConfigControl { constructor(name, options, attr){ super(name, options); this.$numeric = document.createElement("input"); this.$numeric.type = 'number'; this.$numeric.name = name; this.$numeric.value = this.getInitial(); this.$numeric.attr(attr); this.$numeric.addEventListener( 'change', () => this.onchange() ); } get value(){ if( isNaN(this.$numeric.value) ){ this.$numeric.value = this.defaultValue; } return this.$numeric.value; } set value(val){ this.$numeric.value = parseInt(val, 10) || this.defaultValue; } appendTo($to){ $to.appendChild(this.$numeric); } } // コンボボックス管理クラス class ComboSet extends ConfigControl{ constructor(name, options){ super(name, options); this.$combo = document.createElement("select"); this.$combo.attr("name", name); this.$combo.addEventListener( 'change', () => this.onchange() ); if( options.items != null ){ this.addItems(options.items, this.getInitial()); } } get value(){ return this.$combo.value; } set value(val){ this.selectItem(val); } addItems(itemParams, sel = null){ itemParams.forEach( (elm, i) => { let value, text; if( typeof(elm) == "string" ){ value = text = elm; }else{ value = elm.value; text = elm.text || value; } const $item = document.createElement("option"); $item.value = value || "option"+i; $item.textContent = text || $item.value; this.$combo.appendChild($item); } ); if( sel != null ) this.selectItem(sel); } selectItem(target){ if( typeof target == "number" ){ this.$combo.selectedIndex = target; }else if( typeof target == "string" ){ this.$combo.value = target; } } appendTo($to){ $to.appendChild(this.$combo); } // isDefault(){ // if( typeof this.defaultValue == "number" ){ // return this.$combo.selectedIndex == this.defaultValue; // }else if( typeof this.defaultValue == "string" ){ // return this.$combo.value == this.defaultValue; // } // return false; // } } // ラジオボタン管理クラス class RadioSet extends ConfigControl { constructor(name, options){ super(name, options); this.selected_ = null; this.items = []; if( options.items != null ){ this.addItems( options.items, this.getInitial() ); } } get state(){ return this.selected_; } set state(val){ if( val >= 0 && val < this.items.length ){ this.selected_ = val; this.items[val].$radio.checked = true; this.onchange(); } } selectedIndex(){ return this.selected_; } selectedItem(){ let i = this.selectedIndex(); if( i < 0 || i >= this.items.length ){ i = this.defaultValue; } return this.items[i]; } // isDefault(){ // return this.selectedIndex() == this.defaultValue; // } addItems(itemParams, sel = null){ itemParams.forEach( (elm, i) => { const nameItem = `${this.name}-item-${i}`; const forId = elm.id || nameItem; const value = elm.value || nameItem; const title = elm.title || nameItem; const text = elm.text || nameItem; const $label = document.createElement("label"); $label.attr( { 'for': forId, 'title': title, } ); const $radio = document.createElement("input"); $radio.attr( { 'type': "radio", 'name': this.name, 'id': forId, } ); $radio.addEventListener( 'change', () => { this.selected_ = i; this.onchange(); } ); $label.appendChild( $radio ); $label.appendChild( document.createTextNode(text) ); this.items.push( { 'value': value, '$radio': $radio, '$label': $label } ); } ); if( sel != null ) this.state = sel; } appendTo($to){ this.items.forEach( elm => $to.appendChild( elm.$label ) ); } } // コンフィグダイアログ const $configDialog = document.createElement("div"); $configDialog.id = "apc-configDialog"; $configDialog.addEventListener( 'click', () => config.close() ); document.body.appendChild($configDialog); /// コンフィグメニュー const $config = document.createElement("div"); $config.id = "apc-config"; // バブリングストップ $config.addEventListener( 'click', e => e.stopPropagation() ); $configDialog.appendChild($config); const $createH4 = (title, text) => { const $h4 = document.createElement("h4"); $h4.title = title; $h4.textContent = text; return $h4; }; ///config/ 表示する条件 $config.appendChild( $createH4( "APCが表示されるページ", "The page on which Autopagerize_Console is display") ); const radioDisplayRule = new RadioSet( "display_rule", { 'onchange': function(){ $panel.attr(this.name, this.selectedItem().value); }, 'items': [ { 'id': "apc-mdAlways", 'value': "always", 'text': "always", 'title': "常に表示", }, { 'id': "apc-mdValid", 'value': "valid", 'text': "only when \"AutoPagerize\" is valid", 'title': "AutoPagerizeが有効なページであれば表示\n(Alt+Ctrl+pで設定画面を表示)", } ] } ); // radioDisplayRule.isAlways = function(){ // return this.selectedItem() == 0; // }; // radioDisplayRule.isValid = function(){ // return this.selectedItem() == 1; // }; radioDisplayRule.appendTo($config); ///config/position/ マウスが外れている時の表示 $config.appendChild( $createH4( "マウスが外れている時の表示", "Appearance when non-hover" ) ); const radioNonHover = new RadioSet( "non_hover", { 'onchange': function(){ $panel.attr(this.name, this.selectedItem().value); }, 'items': [ { 'id': "apc-no_change", 'value': "no_change", 'text': "No change", 'title': "変更しない", }, { 'id': "apc-transparent", 'value': "transparent", 'text': "Transparent", 'title': "背景を透過", }, { 'id': "apc-stealth", 'value': "stealth", 'text': "Stealth", 'title': "隠れる", } ] } ); radioNonHover.appendTo($config); const $createInputRow = (title, text) => { const $row = document.createElement("div"); $row.className = "apc-mItem"; $row.title = title; $row.textContent = text; return $row; }; const $createGroupBox = subject => { const $box = document.createElement("div"); $box.className = "apc-group"; const $subject = document.createElement("h5"); $subject.textContent = subject; $box.appendChild($subject); return $box; }; /// 位置設定 const UpdatePanelPos = () => { $panel.style[comboPanelPosY.value] = numericPanelPosValue.value + comboPanelPosUnit.value; }; const UpdateBoxPos = ($elm, val) => { $elm.style.height = val+"px"; }; const UpdateAllPositon = () => { UpdatePanelPos(); UpdateBoxPos( $spacingBox12, numericSpecing12.value); UpdateBoxPos( $spacingBox23, numericSpecing23.value); }; ///config/ 位置 $config.appendChild( $createH4( "位置設定", "Expression style") ); //config/position/ パネル位置 const $panelPos = $createInputRow( "パネル位置", "panel position Y:" ); $config.appendChild($panelPos); ///config/position/panelPos/ 使用単位 const comboPanelPosY = new ComboSet( "panel_position_y", { 'items': [ "top", "bottom"], 'onchange': () => { $panel.style.removeProperty("top"); $panel.style.removeProperty("bottom"); UpdatePanelPos(); } } ); comboPanelPosY.appendTo($panelPos); ///config/position/panelPos/ 座標 const numericPanelPosValue = new NumericSet( "panel_position_value", { 'defaultValue': 48, 'onchange': UpdatePanelPos, } ); numericPanelPosValue.appendTo($panelPos); ///config/position/panelPos/ 使用単位 const comboPanelPosUnit = new ComboSet( "panel_position_unit", { 'items': ["px","%"], 'onchange': UpdatePanelPos, } ); comboPanelPosUnit.appendTo($panelPos); ///config/position/ オプションボックス const $optionGroup = $createGroupBox("Option box"); $config.appendChild($optionGroup); ///config/position/optionBox/ 並び順 const $optionOrder = $createInputRow( "並び順", "Order:"); $config.appendChild($optionOrder); const numericOptionOrder = new NumericSet( "option_order", { 'defaultValue': 1, 'onchange': function(){ $optionBox.style.order = this.value*2-1; }, }, { 'min': 1, 'max': 3} ); numericOptionOrder.appendTo($optionOrder); $optionGroup.appendChild($optionOrder); /// 展開方向 const optionDirOptions = { 'items': [ "Left", "Right", "Up", "Down", {value: "Up_protrude", text: "Up (protrude)"}, {value: "Down_protrude", text: "Down (protrude)"} ], 'defaultValue': "Up", 'onchange': function(){ $optionBox.attr(this.name, this.value); } }; ///config/position/optionBox/ 有効ページでの展開方向 const $optionValidForm = $createInputRow( "Autopagerizeが有効なページでの展開方向", "Direction when AP is valid:" ); const comboOptionDirValid = new ComboSet( "option_dir_valid", optionDirOptions ); comboOptionDirValid.appendTo($optionValidForm); $optionGroup.appendChild($optionValidForm); ///config/position/optionBox/ 対象外ページの展開方向 const $optionInvalidForm = $createInputRow( "Autopagerizeが有効でないページでの展開方向", "Direction when AP is invalid:" ); const comboOptionDirInvalid = new ComboSet( "option_dir_invalid", optionDirOptions ); comboOptionDirInvalid.appendTo($optionInvalidForm); $optionGroup.appendChild($optionInvalidForm); ///config/position/scrollerBox/ BOX1とBOX2の間隔 const $specing12 = $createInputRow( "BOX1とBOX2の間隔", "Specing between to BOX1 and BOX2:" ); const numericSpecing12 = new NumericSet( "spacing_12", { 'onchange': function(){ UpdateBoxPos( $spacingBox12, this.value); } } ); numericSpecing12.appendTo($specing12); $specing12.appendChild(document.createTextNode(" px")); $config.appendChild($specing12); ///config/position/ スクローラーボックス const $scrollerGroup = $createGroupBox("Scroller box"); $config.appendChild($scrollerGroup); ///config/position/scrollerBox/ 並び順 const $scrollerOrder = $createInputRow( "並び順", "Order:"); $config.appendChild($scrollerOrder); const numericScrollerOrder = new NumericSet( "scroller_order", { 'defaultValue': 2, 'onchange': function(){ $scrollerBox.style.order = this.value*2-1; } }, { 'min': 1, 'max': 3} ); numericScrollerOrder.appendTo($scrollerOrder); $scrollerGroup.appendChild($scrollerOrder); ///config/position/scrollerBox/ 配置形状 const $scrollerForm = $createInputRow( "配置形状\n(AP対象外ページでは自動的にボタンが2つになるので無視される)", "Form when AP is valid:" ); const comboScrollerForm = new ComboSet( "scroller_form", { 'items': ["Slim","Square"], 'onchange': function(){ $scrollerBox.attr( this.name, this.value); } } ); comboScrollerForm.appendTo($scrollerForm); $scrollerGroup.appendChild($scrollerForm); ///config/position/pageIndexBox/ BOX2とBOX3の間隔 const $specing23 = $createInputRow( " BOX2とBOX3の間隔", "Specing between to BOX2 and BOX3:" ); const numericSpecing23 = new NumericSet( "spacing_23", { 'onchange': function(){ UpdateBoxPos( $spacingBox23, this.value); } } ); numericSpecing23.appendTo($specing23); $specing23.appendChild(document.createTextNode(" px")); $config.appendChild($specing23); ///config/position/ ページインデックスボックス const $pageIndexGroup = $createGroupBox("PageIndex box"); $config.appendChild($pageIndexGroup); ///config/position/pageIndexBox/ 並び順 const $pageIndexOrder = $createInputRow( "並び順", "Order:"); $config.appendChild($pageIndexOrder); const numericPageIndexOrder = new NumericSet( "pageindex_order", { 'defaultValue': 3, 'onchange': function(){ $pageIndexBox.style.order = this.value*2-1; } }, { 'min': 1, 'max': 3} ); numericPageIndexOrder.appendTo($pageIndexOrder); $pageIndexGroup.appendChild($pageIndexOrder); ///config/position/pageIndexBox/ 配置形状 const $pageIndexForm = $createInputRow( "配置形状\n(AP対象外ページでは自動的に非表示となるので無視される)", "Form when AP is valid:" ); const comboPageIndexForm = new ComboSet( "page_index_form", { 'items': ["Pile", "Strip", "Growed Pile"], 'onchange': function(){ $pageIndexBox.attr( this.name, this.value); } } ); comboPageIndexForm.appendTo($pageIndexForm); $pageIndexGroup.appendChild($pageIndexForm); ///config/position/pageIndexBox/ 展開方向 const $pageListExpand = $createInputRow( "展開方向", "Page-list expand direction:" ); const comboPageListExpand = new ComboSet( "page_list_expand", { 'items': [ "Upper", "Lower", "Left-upper", "Left-lower", "Right-upper", "Right-lower" ], 'defaultValue': "Lower", 'onchange': function(){ $pageIndexBox.attr( this.name, this.value); } } ); comboPageListExpand.appendTo($pageListExpand); $pageIndexGroup.appendChild($pageListExpand); const $mButtons = document.createElement("div"); $mButtons.id = "apc-mButtons"; $config.appendChild($mButtons); ///config/ 閉じるボタン const $mCloseDecision = document.createElement("input"); $mCloseDecision.attr( { 'id': "apc-mCloseDecision", 'class': "apc-config_button", 'type': "button", 'value': "OK", } ); $mCloseDecision.addEventListener( 'click', () => config.close() ); $mButtons.appendChild($mCloseDecision); const $mCloseCancel = document.createElement("input"); $mCloseCancel.attr( { 'id': "apc-mCloseCancel", 'class': "apc-config_button", 'type': "button", 'value': "Cancel", } ); $mCloseCancel.addEventListener( 'click', () => config.close(false) ); $mButtons.appendChild($mCloseCancel); class Config { constructor(name, options = null){ this.name = name; this.tempParams = null; this.init(options); } init(options){ if( options == null ) return; if( options.items != null ) this.items = options.items; if( options.onopen != null ) this.onopen_ = options.onopen; if( options.onclose != null ) this.onclose_ = options.onclose; } open(){ this.tempParams = this.getParams(); if( typeof this.onopen_ == "function" ){ this.onopen_(); } } close(decision = true){ if( decision ) this.save(); else this.restore(this.tempParams); this.tempParams = null; if( typeof this.onclose_ == "function" ){ this.onclose_(); } } restore(params){ if( params == null ){ this.items.forEach( elm => elm.setDefault() ); }else{ this.items.forEach( elm => elm.restoreState(params) ); } } getParams(){ const params = {}; this.items.forEach( elm => params[elm.name] = elm.state ); return params; } save(){ GM.setValue( this.name, this.getParams() ); } load(){ GM.getValue( this.name, null).then( result => this.restore(result) ); } } const config = new Config( "config", { 'onopen': () => $panel.setAttribute("config", true), 'onclose': () => { $panel.removeAttribute("config"); UpdateAllPositon(); }, 'items': [ radioDisplayRule, radioNonHover, comboPanelPosY, numericPanelPosValue, comboPanelPosUnit, numericOptionOrder, comboOptionDirValid, comboOptionDirInvalid, numericSpecing12, numericScrollerOrder, comboScrollerForm, numericSpecing23, numericPageIndexOrder, comboPageIndexForm, comboPageListExpand ] } ); const loadState = () => { GM.getValue( apEnable.name, true).then( result => { FireEvent( result? 'AutoPagerizeEnableRequest': 'AutoPagerizeDisableRequest' ); } ); config.load(); UpdateAllPositon(); }; loadState(); // タブを切り替えた時にステータスを最新に同期する window.document.addEventListener( 'visibilitychange', () => { if( window.document.visibilityState == 'visible' ){ loadState(); } } ); })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址