您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Expansion Autopagerize operation. / AutoPagerizeの操作系を拡張します。
// ==UserScript== // @name AutoPagerize_Console // @name:en AutoPagerize_Console // @name:ja AutoPagerize_Console // @description Expansion Autopagerize operation. / AutoPagerizeの操作系を拡張します。 // @description:en Expansion Autopagerize operation. // @namespace phodra // @include * // @exclude https://www.youtube.com/* // @exclude https://docs.google.com/* // @exclude https://console.developers.google.com/* // @version 5.4.1 // @noframes // @grant GM_getValue // @grant GM_setValue // @grant GM.getValue // @grant GM.setValue // ==/UserScript== (()=>{ // Greasemonkey 4.0未満バージョン対応 if( this.GM == null ){ this.GM = { getValue: async (name, defVal) => await GM_getValue(name, defVal), setValue: async (name, value) => await 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, #apc-mButtons { background-color: #1c1c1c !important; } .apc-button { filter: brightness(100%) contrast(150%); } #apc-enabler { filter: grayscale(100%); } #apc-panel[ap_valid] #apc-enabler { filter: grayscale(60%); } /* コンソールパネル */ #apc-panel { opacity: 0.6; background-color: transparent !important; position: fixed; z-index: 9999999990; right: 0; margin: 0; display: inline-flex; flex-direction: column; align-items: flex-end; } #apc-panel[display_rule=valid]:not([ap_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[ap_valid] #apc-optionBox[option_dir_valid='Left'], #apc-panel:not([ap_valid]) #apc-optionBox[option_dir_invalid='Left'] { flex-direction: row-reverse; } #apc-panel[ap_valid] #apc-optionBox[option_dir_valid='Right'], #apc-panel:not([ap_valid]) #apc-optionBox[option_dir_invalid='Right'] { flex-direction: row; } #apc-panel[ap_valid] #apc-optionBox[option_dir_valid^='Up'], #apc-panel:not([ap_valid]) #apc-optionBox[option_dir_invalid^='Up'] { flex-direction: column-reverse; } #apc-panel[ap_valid] #apc-optionBox[option_dir_valid^='Down'], #apc-panel:not([ap_valid]) #apc-optionBox[option_dir_invalid^='Down'] { flex-direction: column; } #apc-panel[ap_valid] #apc-optionBox[option_dir_valid='Up_protrude'], #apc-panel:not([ap_valid]) #apc-optionBox[option_dir_invalid='Up_protrude'] { margin-top: -18px; } #apc-panel[ap_valid] #apc-optionBox[option_dir_valid='Down_protrude'], #apc-panel:not([ap_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([ap_valid]) #apc-scrollBottom { order: 4; transform: scale(1, -1); } /* scrollerBox配置形状:スクエア */ #apc-panel[ap_valid] #apc-scrollerBox[scroller_form='Square'] { display: grid; } #apc-panel[ap_valid] #apc-scrollerBox[scroller_form='Square'] #apc-scrollTop { grid-row: 1/2; grid-column: 2/3; transform: scale(-1, 1); } #apc-panel[ap_valid] #apc-scrollerBox[scroller_form='Square'] #apc-scrollBottom { grid-row: 2/3; grid-column: 2/3; transform: scale(-1, -1); } #apc-panel[ap_valid] #apc-scrollerBox[scroller_form='Square'] #apc-scrollNext { grid-row: 2/3; grid-column: 1/2; transform: scale(1, -1); } #apc-panel[ap_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([ap_valid]) #apc-scrollPrev, #apc-panel:not([ap_valid]) #apc-scrollNext, #apc-panel:not([ap_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; padding-bottom: 0; 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; margin: 0; margin-top: 10px; padding: 5px; font-size: 0; } #apc-config .apc-group h5 { 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 { 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; padding: 15px 0 0 0; position: sticky; bottom: 0; } #apc-config .apc-config_button { border: outset 2px; margin: 0 5px; width: 100%; } ` document.head.appendChild($style) // コントロール配置 /// パネル(最親) const $panel = document.createElement("div") $panel.id = "apc-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, } ///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) ) document.body.appendChild($panel) /// コンフィグメニューを作成 // コンフィグコントロール管理クラス // 基底クラス 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 $config = document.createElement("div") $config.id = "apc-config" // バブリングストップ $config.addEventListener( 'click', e => e.stopPropagation() ) 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) // コンフィグダイアログ const $configDialog = document.createElement("div") $configDialog.id = "apc-configDialog" $configDialog.addEventListener( 'click', () => config.close() ) $configDialog.appendChild($config) document.body.appendChild($configDialog) 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 ] } ) // enableの変更を反映 const updateEnable = (toggle = null, save = true) => { apEnable.state = toggle === null ? !apEnable.state : toggle $enabler.attr( apEnable.state ? { 'state': "enable", 'alt': "E", 'title': "AutePagerize OFF (Now:Enable)" } : { 'state': "disable", 'alt': "D", 'title': "AutePagerize ON (Now:Disable)" } ) if( save ){ GM.setValue( apEnable.name, apEnable.state) } } ///panel/optionBox// APイベントをキャプチャー document.addEventListener( 'AutoPagerizeToggleRequest', () => updateEnable() ) document.addEventListener( 'AutoPagerizeEnableRequest', () => updateEnable(true, false) ) document.addEventListener( 'AutoPagerizeDisableRequest', () => updateEnable(false, false) ) // リクエストを発行 const RequestEnableOrDisable = enable => { FireEvent( enable ? 'AutoPagerizeEnableRequest' : 'AutoPagerizeDisableRequest' ) } // すべての状態をロードする const loadState = () => { GM.getValue( apEnable.name, true) .then( result => { apEnable.state = result RequestEnableOrDisable(result) } ) config.load() UpdateAllPositon() } loadState() // タブを切り替えた時にステータスを最新に同期する window.document.addEventListener( 'visibilitychange', () => { if( window.document.visibilityState == 'visible' ){ loadState() } } ) // AutoPagerize有効なページでのみ行う処理 const updateApValid = () => { setTimeout( () => { RequestEnableOrDisable(apEnable.state) }, 100 ) $panel.attr( "ap_valid", true) } /// APメッセージバー(アドオン版)かAPアイコン(スクリプト版)が /// 存在する、もしくは挿入されると、APが有効なページであるとみなす。 const apObject = document.getElementById("autopagerize_message_bar") || document.getElementById("autopagerize_icon") if( apObject != null ){ updateApValid() }else{ const mo = new MutationObserver( mRecs => { for( let rec of mRecs ){ for( let $added of rec.addedNodes ){ if( ["autopagerize_message_bar", "autopagerize_icon"] .includes($added.id) ) { updateApValid() 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) } })()
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址