AutoPagerize_Console

AutoPagerizeの操作系を拡張します。

当前为 2018-05-20 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name AutoPagerize_Console
  3. // @namespace phodra
  4. // @description AutoPagerizeの操作系を拡張します。
  5. // @include *
  6. // @version 5.00
  7. // @noframes
  8. // @grant GM_getValue
  9. // @grant GM_setValue
  10. // @grant GM.getValue
  11. // @grant GM.setValue
  12. // ==/UserScript==
  13.  
  14. (()=>{
  15. // Greasemonkey 4.0未満バージョン対応
  16. if( this.GM == null ){
  17. this.GM = {};
  18. this.GM.getValue = async (name, defaultValue) => {
  19. return GM_getValue(name, defaultValue);
  20. };
  21. this.GM.setValue = async (name, value) => {
  22. return GM_setValue(name, value);
  23. };
  24. }
  25. // 拡張
  26. Array.prototype.first = function(){ return this[0]; };
  27. Array.prototype.last = function(){ return this[this.length-1]; };
  28. Array.prototype.round = function(i){ return this[i<0? 0: i>=this.length? this.length-1: i]; };
  29. Node.prototype.prependChild = function(elm){ this.insertBefore(elm,this.firstChild); };
  30. Element.prototype.css = function(arg1, arg2){
  31. if( arg2 == null){
  32. for( let key in arg1 ){
  33. this.style[key] = arg1[key];
  34. }
  35. }else{
  36. this.style[arg1] = arg2;
  37. }
  38. };
  39. Element.prototype.attr = function(arg1, arg2){
  40. if( arg2 == null){
  41. for( let key in arg1 ){
  42. this.setAttribute(key, arg1[key]);
  43. }
  44. }else{
  45. this.setAttribute(arg1, arg2);
  46. }
  47. };
  48. // 任意のイベントを発火
  49. const FireEvent = ename => {
  50. const e = document.createEvent('Event');
  51. e.initEvent( ename, true, false);
  52. return document.dispatchEvent(e);
  53. };
  54.  
  55.  
  56. // 画像をひとまとめにしておく
  57. const RES = {
  58. 'config': "",
  59. 'scrAll': "",
  60. 'scrPage': "",
  61. 'disabled': "",
  62. 'enabled': "",
  63. };
  64.  
  65.  
  66.  
  67. // スタイル
  68. const $style = document.createElement("style");
  69. $style.setAttribute("type", "text/css");
  70. $style.textContent = `
  71. #apc-panel *,
  72. #apc-config * {
  73. color: #eee;
  74. border-color: #666;
  75. }
  76. .apc-box,
  77. #apc-pageList,
  78. #apc-config {
  79. background-color: #1c1c1c !important;
  80. }
  81. .apc-button {
  82. filter: brightness(100%) contrast(150%);
  83. }
  84. #apc-enabler {
  85. filter: grayscale(100%);
  86. }
  87. #apc-panel[valid] #apc-enabler {
  88. filter: grayscale(60%);
  89. }
  90.  
  91.  
  92. /* コンソールパネル */
  93. #apc-panel {
  94. opacity: 0.6;
  95. background-color: transparent !important;
  96.  
  97. position: fixed;
  98. z-index: 9999999990;
  99. right: 0;
  100. /*top: 48px;*/
  101. margin: 0;
  102.  
  103. display: inline-flex;
  104. flex-direction: column;
  105. align-items: flex-end;
  106.  
  107. }
  108. #apc-panel[display_rule=valid]:not([valid]) {
  109. visibility: hidden;
  110. }
  111.  
  112. #apc-panel *,
  113. #apc-config * {
  114. font-family: arial, sans-serif;
  115. font-size: 14px;
  116. font-weight: normal;
  117. letter-spacing: normal;
  118. vertical-align: baseline;
  119. line-height: normal;
  120. }
  121. .apc-box {
  122. padding: 1px;
  123. box-sizing: content-box;
  124. min-width: 18px;
  125. /*position: relative;*/
  126. pointer-events: auto;
  127. }
  128. #apc-panel[non_hover='transparent']:not(:hover):not([config]) .apc-box {
  129. background-color: transparent !important;
  130. }
  131. #apc-panel[non_hover='stealth']:not(:hover):not([config]) .apc-box {
  132. visibility: hidden;
  133. }
  134.  
  135. /* ボタンのスタイル */
  136. .apc-button {
  137. border: solid 1px;
  138. box-sizing: border-box;
  139. cursor: pointer;
  140. margin: 1px;
  141. padding: 0;
  142. position: relative;
  143. opacity: 0.5;
  144. width: 16px;
  145. }
  146. .apc-button:hover {
  147. opacity: 0.8;
  148. }
  149. .apc-button:active {
  150. opacity: 0.3;
  151. }
  152. /* ボタンの大きさ */
  153. #apc-optionBox .apc-button {
  154. height: 16px;
  155. }
  156. #apc-scrollerBox .apc-button {
  157. height: 32px;
  158. }
  159.  
  160.  
  161. /* オプションボックス */
  162. #apc-optionBox {
  163. display: flex;
  164. z-index: 9999999991;
  165. }
  166. #apc-enabler[state="enable"] {
  167. background-image: url(${RES.enabled});
  168. }
  169. #apc-enabler[state="disable"] {
  170. background-image: url(${RES.disabled});
  171. }
  172. #apc-setting {
  173. background-image: url(${RES.config});
  174. }
  175. /* optionBox配置形状: */
  176. #apc-panel[valid] #apc-optionBox[option_dir_valid='Left'],
  177. #apc-panel:not([valid]) #apc-optionBox[option_dir_invalid='Left'] {
  178. flex-direction: row-reverse;
  179. }
  180. #apc-panel[valid] #apc-optionBox[option_dir_valid='Right'],
  181. #apc-panel:not([valid]) #apc-optionBox[option_dir_invalid='Right'] {
  182. flex-direction: row;
  183. }
  184. #apc-panel[valid] #apc-optionBox[option_dir_valid^='Up'],
  185. #apc-panel:not([valid]) #apc-optionBox[option_dir_invalid^='Up'] {
  186. flex-direction: column-reverse;
  187. }
  188. #apc-panel[valid] #apc-optionBox[option_dir_valid^='Down'],
  189. #apc-panel:not([valid]) #apc-optionBox[option_dir_invalid^='Down'] {
  190. flex-direction: column;
  191. }
  192. #apc-panel[valid] #apc-optionBox[option_dir_valid='Up_protrude'],
  193. #apc-panel:not([valid]) #apc-optionBox[option_dir_invalid='Up_protrude'] {
  194. margin-top: -18px;
  195. }
  196. #apc-panel[valid] #apc-optionBox[option_dir_valid='Down_protrude'],
  197. #apc-panel:not([valid]) #apc-optionBox[option_dir_invalid='Down_protrude'] {
  198. margin-bottom: -18px;
  199. }
  200.  
  201.  
  202. /* スクローラーボックス */
  203. #apc-scrollerBox {
  204. z-index: 9999999992;
  205. }
  206. #apc-scrollTop,
  207. #apc-scrollBottom {
  208. background-image: url(${RES.scrAll});
  209. }
  210. #apc-scrollPrev,
  211. #apc-scrollNext {
  212. background-image: url(${RES.scrPage});
  213. }
  214. /* scrollerBox配置形状:スリム */
  215. #apc-scrollerBox[scroller_form='Slim'] {
  216. display: flex;
  217. flex-flow: column wrap-reverse;
  218. }
  219. #apc-scrollerBox[scroller_form='Slim'] #apc-scrollTop {
  220. order: 1;
  221. }
  222. #apc-scrollerBox[scroller_form='Slim'] #apc-scrollPrev {
  223. order: 2;
  224. }
  225. #apc-scrollerBox[scroller_form='Slim'] #apc-scrollNext {
  226. order: 3;
  227. transform: scale(1, -1);
  228. }
  229. #apc-scrollerBox[scroller_form='Slim'] #apc-scrollBottom,
  230. #apc-panel:not([valid]) #apc-scrollBottom {
  231. order: 4;
  232. transform: scale(1, -1);
  233. }
  234. /* scrollerBox配置形状:スクエア */
  235. #apc-panel[valid] #apc-scrollerBox[scroller_form='Square'] {
  236. display: grid;
  237. }
  238. #apc-panel[valid] #apc-scrollerBox[scroller_form='Square'] #apc-scrollTop {
  239. grid-row: 1/2;
  240. grid-column: 2/3;
  241. transform: scale(-1, 1);
  242. }
  243. #apc-panel[valid] #apc-scrollerBox[scroller_form='Square'] #apc-scrollBottom {
  244. grid-row: 2/3;
  245. grid-column: 2/3;
  246. transform: scale(-1, -1);
  247. }
  248. #apc-panel[valid] #apc-scrollerBox[scroller_form='Square'] #apc-scrollNext {
  249. grid-row: 2/3;
  250. grid-column: 1/2;
  251. transform: scale(1, -1);
  252. }
  253. #apc-panel[valid] #apc-scrollerBox[scroller_form='Square'] #apc-scrollPrev {
  254. grid-row: 1/2;
  255. grid-column: 1/2;
  256. }
  257.  
  258.  
  259. /* ページボックス */
  260. #apc-pageIndexBox {
  261. cursor: pointer;
  262. z-index: 9999999993;
  263. min-height: 17px;
  264. position: relative;
  265. }
  266. /* ページ表示 */
  267. #apc-sequencer {
  268. margin: 0px 1px;
  269. padding: 0px;
  270. text-align: center;
  271. }
  272. /* pageIndexBox配置形状:縦置き */
  273. #apc-pageIndexBox[page_index_form$='Pile'] #apc-sequencer {
  274. display: flex;
  275. flex-direction: column;
  276. }
  277. #apc-pageIndexBox[page_index_form$='Pile'] span:first-child{
  278. border-bottom: solid 1px #eee !important;
  279. }
  280. /* pageIndexBox配置形状:横置き */
  281. #apc-pageIndexBox[page_index_form='Strip'],
  282. #apc-pageIndexBox[page_index_form='Growed Pile']{
  283. box-sizing: border-box;
  284. width: 100%;
  285. }
  286. #apc-pageIndexBox[page_index_form$='Strip'] span:first-child::after{
  287. content: " / ";
  288. }
  289.  
  290. /* ページ一覧 */
  291. #apc-pageList {
  292. position: absolute;
  293. min-width: 38px;
  294. margin: 0;
  295. padding: 0px;
  296. list-style-type: none;
  297.  
  298. display: none;
  299. flex-flow: column wrap-reverse;
  300. max-height: 50vh;
  301. }
  302. #apc-pageIndexBox:hover #apc-pageList {
  303. display: flex;
  304. }
  305. #apc-pageIndexBox[page_list_expand^="Left"] #apc-pageList {
  306. right: 100%;
  307. }
  308. #apc-pageIndexBox[page_list_expand^="Right"] #apc-pageList {
  309. left: 100%;
  310. flex-wrap: wrap;
  311. }
  312. #apc-pageIndexBox[page_list_expand$="upper"] #apc-pageList {
  313. bottom: 0;
  314. }
  315. #apc-pageIndexBox[page_list_expand$="lower"] #apc-pageList {
  316. top: 0;
  317. }
  318. #apc-pageIndexBox[page_list_expand="Upper"] #apc-pageList {
  319. right: 0;
  320. bottom: 100%;
  321. }
  322. #apc-pageIndexBox[page_list_expand="Lower"] #apc-pageList {
  323. right: 0;
  324. top: 100%;
  325. }
  326. .apc-pageListItem {
  327. border-style: outset;
  328. border-width: 1px;
  329. box-sizing: border-box;
  330. cursor: pointer;
  331. margin: 1px;
  332. padding: 0px;
  333. text-align: center;
  334. }
  335.  
  336. #apc-panel:not([valid]) #apc-scrollPrev,
  337. #apc-panel:not([valid]) #apc-scrollNext,
  338. #apc-panel:not([valid]) #apc-pageIndexBox {
  339. display: none;
  340. }
  341.  
  342. #apc-panel[config]+#apc-configDialog {
  343. display: flex;
  344. }
  345. /* コンフィグメニューダイアログ */
  346. #apc-configDialog {
  347. background-color: transparent !important;
  348. position: fixed;
  349. z-index: 9999999999;
  350. top: 0px;
  351. left: 0px;
  352. width: 100%;
  353. height: 100%;
  354. display: none;
  355. flex-direction: column;
  356. justify-content: center;
  357. align-items: center;
  358. }
  359. #apc-config {
  360. opacity: 1.0;
  361. border: outset 3px white;
  362. display: inline-block;
  363.  
  364. text-align: left;
  365. padding: 5px 15px;
  366. max-height: 90vh;
  367. overflow: scroll;
  368.  
  369. user-select: none;
  370. -moz-user-select: none;
  371. -webkit-user-select: none;
  372. -ms-user-select: none;
  373. }
  374. #apc-config h4 {
  375. display: block;
  376. background-color: transparent;
  377. padding: 0;
  378. padding-bottom: 1px;
  379. margin: 5px 0 2px -5px;
  380. border: solid 0;
  381. border-bottom-width: 1px;
  382. font-weight: bold !important;
  383. }
  384. #apc-config .apc-group {
  385. border: solid 1px;
  386. /*position: relative;*/
  387. margin: 0;
  388. margin-top: 10px;
  389. padding: 5px;
  390. font-size: 0;
  391. }
  392. #apc-config .apc-group h5 {
  393. /*position: relative;*/
  394. display: inline-block;
  395. margin: 0;
  396. margin-top: -1em;
  397. padding: 0 3px;
  398. background-color: #1c1c1c;
  399. font-weight: bold !important;
  400. }
  401. #apc-config .apc-mItem {
  402. display: block;
  403. margin: 5px 2px 0 2px;
  404. }
  405. #apc-config label {
  406. display: block;
  407. margin: 2px;
  408. cursor: pointer;
  409. }
  410. #apc-config input,
  411. #apc-config select {
  412. /*color: white;*/
  413. background-image: none;
  414. background-color: #000;
  415. border: 2px inset #666;
  416. margin: auto 2px;
  417. cursor: pointer;
  418. border-radius: initial;
  419. padding: initial;
  420. }
  421. #apc-config option {
  422. color: black;
  423. background-color: white;
  424. }
  425. #apc-config input {
  426. width: auto;
  427. height: auto;
  428. padding: 0;
  429. cursor: pointer;
  430. }
  431. #apc-config select,
  432. #apc-config input[type='number'] {
  433. box-sizing: content-box;
  434. height: 19px;
  435. }
  436. #apc-config input[type='number'] {
  437. max-width: 60px;
  438. -webkit-appearance: ;
  439. }
  440. #apc-config select {
  441. max-width: 80px;
  442. }
  443. #apc-config input[type='checkbox'] {
  444. vertical-align: middle;
  445. }
  446. #apc-config #apc-mButtons {
  447. display: flex;
  448. justify-content: stretch;
  449. margin: 10px 0 5px 0;
  450. }
  451. #apc-config .apc-config_button {
  452. border: outset 2px;
  453. margin: 0 5px;
  454. width: 100%;
  455. }
  456. #apc-config .apc-config_button:first-child {
  457. margin-left: 0;
  458. }
  459. #apc-config .apc-config_button:last-child {
  460. margin-right: 0;
  461. }
  462. `;
  463. document.head.appendChild($style);
  464.  
  465.  
  466.  
  467. // APが有効なページであるのか
  468. const apValid = {
  469. name: "valid",
  470. flag: null,
  471. };
  472. /// APメッセージバー(アドオン版)かAPアイコン(スクリプト版)が
  473. /// 挿入されると、APが有効なページであるとみなす。
  474. const checkApValid = $elm => {
  475. if( apValid.flag == null ){
  476. // AutoPagerize有効なページでのみ行う処理
  477. if( $elm.id == "autopagerize_message_bar" ||
  478. $elm.id == "autopagerize_icon" )
  479. {
  480. apValid.flag = $elm.id;
  481. $panel.attr( apValid.name, true);
  482. return true;
  483. }
  484. }
  485. return false;
  486. };
  487. const apObject = document.getElementById("autopagerize_message_bar") ||
  488. document.getElementById("autopagerize_icon");
  489. if( apObject != null ){
  490. checkApValid(apObject);
  491. }else{
  492. const mo = new MutationObserver(
  493. mRecs => {
  494. for( let rec of mRecs ){
  495. for( let added of rec.addedNodes ){
  496. if( checkApValid(added) ){
  497. mo.disconnect();
  498. }
  499. }
  500. }
  501. }
  502. );
  503. mo.observe( document.body, {'childList': true});
  504. document.addEventListener(
  505. 'GM_AutoPagerizeNextPageLoaded',
  506. () => mo.disconnect()
  507. );
  508. }
  509.  
  510. let scrTop = document.documentElement.scrollTop;
  511. let oldPage = null;
  512. // ウィンドウのスクロールが発生した時
  513. window.addEventListener(
  514. 'scroll',
  515. () => {
  516. scrTop = document.documentElement.scrollTop;
  517. // 現在のページを更新
  518. const np = getNowPage();
  519. if( np != oldPage ){
  520. $sequencerNow.textContent = np+1;
  521. oldPage = np;
  522. }
  523. }
  524. );
  525. // 各ページを管理する配列
  526. let pageBounds = [0];
  527. // 現在のページを取得(境界線上を前ページとみなす場合false)
  528. const getNowPage = (notLess=true) => {
  529. const breakOver = notLess ?
  530. (st, bound) => st >= bound :
  531. (st, bound) => st > bound ;
  532. let i;
  533. for( i = pageBounds.length-1; i >= 0; i-- ){
  534. if( breakOver( scrTop, pageBounds[i]) ) break;
  535. }
  536. return i;
  537. };
  538. const getBottomPos = () => {
  539. return Math.max(
  540. document.body.clientHeight,
  541. document.body.scrollHeight,
  542. document.documentElement.scrollHeight,
  543. document.documentElement.clientHeight) -
  544. window.innerHeight;
  545. };
  546.  
  547. // ページを継ぎ足した時、継ぎ目の位置を記録する
  548. const apPageAppended = () => {
  549. // セパレーターの絶対位置を取得
  550. const $apSep = document.getElementsByClassName("autopagerize_page_separator");
  551. const len = $apSep.length;
  552.  
  553. let sepPos = $apSep.item(len-1).getBoundingClientRect().top + window.pageYOffset;
  554. sepPos = Math.round(sepPos);
  555. if( !pageBounds.includes(sepPos) ){
  556. pageBounds.push(sepPos);
  557. }
  558. $sequencerMax.textContent = len+1;
  559.  
  560. // ページリストアイテムを追加
  561. $pageList.appendChild( $createPageListItem(len+1) );
  562. };
  563.  
  564. document.addEventListener(
  565. 'GM_AutoPagerizeNextPageLoaded',
  566. () => apPageAppended()
  567. );
  568.  
  569. let scrTick = null;
  570. // 任意の位置にスクロール
  571. const smoothScrollTo = targetY => {
  572. // 続けてスクロールさせるとキャンセルして新たなスクロール
  573. if( scrTick != null ){
  574. clearTimeout(scrTick);
  575. scrTick = null;
  576. }
  577. // 再帰処理部分
  578. const smoothScrollTick = (_targetY) => {
  579. // 常に最新の位置をターゲットにしたい場合、引数に関数を渡す
  580. // (最下部へのスクロール用。
  581. //  なぜかスクロール途中にページの高さが変わる場合がある)
  582. let targetY = (typeof(_targetY) == "function" ?
  583. _targetY() :
  584. _targetY);
  585. // 移動量
  586. let moveY = (targetY-scrTop)/10;
  587. // 終了条件
  588. let endOver;
  589. // 下方向の移動量調整と終了条件
  590. if( moveY>0 ){
  591. moveY = Math.ceil(moveY); //切り上げ
  592. endOver = scrTop+moveY>=targetY; //ターゲットを越えていれば
  593. }
  594. // 上方向の移動量調整と終了条件
  595. else{
  596. moveY = Math.floor(moveY); //切り下げ
  597. endOver = scrTop+moveY<=targetY; //ターゲットを下回っていれば
  598. }
  599. // 条件を満たしていれば終了
  600. if( endOver ){
  601. window.scrollTo( 0, targetY);
  602. clearTimeout(scrTick);
  603. scrTick = null;
  604. }else{
  605. window.scrollBy( 0, moveY);
  606. scrTick = setTimeout(
  607. () => smoothScrollTick(targetY),
  608. 10
  609. );
  610. }
  611. };
  612. smoothScrollTick(targetY);
  613. };
  614.  
  615.  
  616.  
  617. // コントロール配置
  618. /// パネル(最親)
  619. const $panel = document.createElement("div");
  620. $panel.id = "apc-panel";
  621. document.body.appendChild($panel);
  622.  
  623. // ボックスを作成
  624. const $createBox = () => {
  625. const $box = document.createElement("div");
  626. $box.className = "apc-box";
  627. return $box;
  628. };
  629. // ボタンを作成
  630. const $createButton = (attr) => {
  631. const $button = document.createElement("div");
  632. $button.className = "apc-button";
  633. $button.attr(attr);
  634. return $button;
  635. };
  636.  
  637. ///panel/ オプションボックス
  638. const $optionBox = $createBox();
  639. $optionBox.id = "apc-optionBox";
  640. $panel.appendChild($optionBox);
  641.  
  642. ///panel/optionBox/ オンオフボタン (enable/disable)
  643. const $enabler = $createButton(
  644. { 'id': "apc-enabler" }
  645. );
  646. const apEnable = {
  647. name: "enable",
  648. state: true,
  649. };
  650. const changeToggle = (toggle = null) => {
  651. apEnable.state = toggle != null ? toggle : !apEnable.state;
  652. $enabler.attr(
  653. apEnable.state ?
  654. {
  655. 'state': "enable",
  656. 'alt': "E",
  657. 'title': "AutePagerize OFF (Now:Enable)"
  658. } :
  659. {
  660. 'state': "disable",
  661. 'alt': "D",
  662. 'title': "AutePagerize ON (Now:Disable)"
  663. }
  664. );
  665. GM.setValue( apEnable.name, apEnable.state);
  666. };
  667. ///panel/optionBox// APイベントをキャプチャー
  668. document.addEventListener(
  669. 'AutoPagerizeToggleRequest', () => changeToggle()
  670. );
  671. document.addEventListener(
  672. 'AutoPagerizeEnableRequest', () => changeToggle(true)
  673. );
  674. document.addEventListener(
  675. 'AutoPagerizeDisableRequest', () => changeToggle(false)
  676. );
  677. ///panel/optionBox// トグルボタン クリックでリクエストイベント発火
  678. $enabler.addEventListener(
  679. 'click', () => FireEvent('AutoPagerizeToggleRequest')
  680. );
  681. $optionBox.appendChild($enabler);
  682.  
  683. // コンフィグメニューを開いているか
  684. ///panel/optionBox/ 設定ボタン
  685. const $setting = $createButton(
  686. {
  687. 'id': "apc-setting",
  688. 'title': "Open Config",
  689. 'alt': "c",
  690. }
  691. );
  692. $optionBox.appendChild($setting);
  693. $setting.addEventListener(
  694. 'click', () => config.open()
  695. );
  696. /// ボタンが見えなくなった時のため、ショートカットで開けるようにする。
  697. /// Alt + Ctrl + p
  698. document.addEventListener(
  699. 'keydown', e => {
  700. if( e.altKey && e.ctrlKey && e.keyCode == 80 ){
  701. config.open();
  702. }
  703. }
  704. );
  705.  
  706. const $spacingBox12 = document.createElement("div");
  707. $spacingBox12.className = "apc-spacing";
  708. $spacingBox12.style.order = 2;
  709. $panel.appendChild($spacingBox12);
  710.  
  711. ///panel/ スクロールボックス
  712. const $scrollerBox = $createBox();
  713. $scrollerBox.id = "apc-scrollerBox";
  714. $panel.appendChild($scrollerBox);
  715.  
  716. ///panel/scr/ 最上部へ移動
  717. const $scrollTop = $createButton(
  718. {
  719. 'id': "apc-scrollTop",
  720. 'alt': "↑",
  721. 'title': "Move to Top",
  722. }
  723. );
  724. $scrollerBox.appendChild($scrollTop);
  725. ///panel/scr/ 最下部へ移動
  726. const $scrollBottom = $createButton(
  727. {
  728. 'id': "apc-scrollBottom",
  729. 'alt': "↓",
  730. 'title': "Move to Bottom",
  731. }
  732. );
  733. $scrollerBox.appendChild($scrollBottom);
  734. ///panel/scr/ 前のページ
  735. const $scrollPrev = $createButton(
  736. {
  737. 'id': "apc-scrollPrev",
  738. 'alt': "△",
  739. 'title': "Move to Previous",
  740. }
  741. );
  742. $scrollerBox.appendChild($scrollPrev);
  743. ///panel/scr/ 次のページ
  744. const $scrollNext = $createButton(
  745. {
  746. 'id': "apc-scrollNext",
  747. 'alt': "▽",
  748. 'title': "Move to Next",
  749. }
  750. );
  751. $scrollerBox.appendChild($scrollNext);
  752.  
  753. ///panel/scr/ 最上部へ移動
  754. $scrollTop.addEventListener(
  755. 'click', () => smoothScrollTo(0)
  756. );
  757. ///panel/scr/ 最下部へ移動
  758. $scrollBottom.addEventListener(
  759. 'click', () => smoothScrollTo(getBottomPos)
  760. );
  761. ///panel/scr/ 前のページ
  762. $scrollPrev.addEventListener(
  763. 'click', () => {
  764. smoothScrollTo( pageBounds.round( getNowPage(false) ) );
  765. }
  766. );
  767. ///panel/scr/ 次のページ
  768. $scrollNext.addEventListener(
  769. 'click', () => {
  770. const targetPage = getNowPage()+1;
  771. smoothScrollTo(
  772. targetPage < pageBounds.length ?
  773. pageBounds.round(targetPage) :
  774. getBottomPos
  775. );
  776. }
  777. );
  778.  
  779. const $spacingBox23 = document.createElement("div");
  780. $spacingBox23.className = "apc-spacing"
  781. $spacingBox23.style.order = 4;
  782. $panel.appendChild($spacingBox23);
  783.  
  784. ///panel/ ページボックス
  785. const $pageIndexBox = $createBox();
  786. $pageIndexBox.id = "apc-pageIndexBox";
  787.  
  788. ///panel/pageIndexBox/ ページ数表示
  789. const $sequencer = document.createElement("div");
  790. $sequencer.id = "apc-sequencer";
  791. const $sequencerNow = document.createElement("span");
  792. const $sequencerMax = document.createElement("span");
  793. $sequencerNow.textContent = $sequencerMax.textContent = "1";
  794. $sequencer.appendChild($sequencerNow);
  795. $sequencer.appendChild($sequencerMax);
  796. $pageIndexBox.appendChild($sequencer);
  797. $panel.appendChild($pageIndexBox);
  798.  
  799. ///panel/pageIndexBox/ ページリスト
  800. const $pageList = document.createElement("ol");
  801. $pageList.id = "apc-pageList";
  802. $pageIndexBox.appendChild($pageList);
  803. // 新しいページリストアイテムにイベントを追加
  804. const $createPageListItem = num => {
  805. const $elm = document.createElement("li");
  806. $elm.className = "apc-pageListItem";
  807. $elm.textContent = num;
  808. // クリックでそのページにスクロール
  809. $elm.addEventListener(
  810. 'click', function(){
  811. const targetNum = this.textContent-1;
  812. smoothScrollTo( pageBounds.round(targetNum) );
  813. }
  814. );
  815. // // ダブルクリックでページ移動
  816. // $elm.addEventListener(
  817. // 'dblclick', function(){
  818. // const num = this.textContent-2;
  819. // if( num >= 0 )
  820. // document.getElementsByClassName("autopagerize_link")[num].href;
  821. // }
  822. // );
  823. return $elm;
  824. };
  825. $pageList.appendChild( $createPageListItem(1) );
  826.  
  827.  
  828.  
  829.  
  830.  
  831.  
  832. /// コンフィグメニューを作成
  833. // コンフィグコントロール管理クラス
  834. // 基底クラス
  835. class ConfigControl {
  836. constructor(name, options){
  837. this.name = name;
  838. this.defaultValue = options.defaultValue || 0;
  839. this.onchange_ = options.onchange || null;
  840. }
  841.  
  842. get state(){
  843. return this.value;
  844. }
  845. set state(val){
  846. this.value = val;
  847. this.onchange();
  848. }
  849. onchange(){
  850. if( typeof(this.onchange_) == "function" ) this.onchange_();
  851. }
  852. getInitial(){
  853. return this.defaultValue || 0;
  854. }
  855. setDefault(){
  856. this.state = this.defaultValue;
  857. }
  858. restoreState(params){
  859. const val = params[this.name];
  860. if( val == null ){
  861. this.setDefault();
  862. }else{
  863. this.state = val;
  864. }
  865. }
  866. /* 仮想関数
  867. * 継承先のクラスでこれらをオーバーライドする
  868. * オーバーライドされていなければバグなのでwarn
  869. */
  870. /* virtual */get value(){
  871. console.warn("virtual get value", this.name);
  872. return null;
  873. }
  874. /* virtual */set value(val){
  875. console.warn("virtual set value", this.name);
  876. }
  877. /* virtual */appendTo($to){
  878. console.warn("virtual appendTo", this.name);
  879. }
  880. }
  881. // NumericUpDown管理クラス
  882. class NumericSet extends ConfigControl {
  883. constructor(name, options, attr){
  884. super(name, options);
  885.  
  886. this.$numeric = document.createElement("input");
  887. this.$numeric.type = 'number';
  888. this.$numeric.name = name;
  889. this.$numeric.value = this.getInitial();
  890. this.$numeric.attr(attr);
  891. this.$numeric.addEventListener(
  892. 'change', () => this.onchange()
  893. );
  894. }
  895. get value(){
  896. if( isNaN(this.$numeric.value) ){
  897. this.$numeric.value = this.defaultValue;
  898. }
  899. return this.$numeric.value;
  900. }
  901. set value(val){
  902. this.$numeric.value = parseInt(val, 10) || this.defaultValue;
  903. }
  904. appendTo($to){
  905. $to.appendChild(this.$numeric);
  906. }
  907. }
  908. // コンボボックス管理クラス
  909. class ComboSet extends ConfigControl{
  910. constructor(name, options){
  911. super(name, options);
  912.  
  913. this.$combo = document.createElement("select");
  914. this.$combo.attr("name", name);
  915. this.$combo.addEventListener(
  916. 'change', () => this.onchange()
  917. );
  918.  
  919. if( options.items != null ){
  920. this.addItems(options.items, this.getInitial());
  921. }
  922.  
  923. }
  924. get value(){
  925. return this.$combo.value;
  926. }
  927. set value(val){
  928. this.selectItem(val);
  929. }
  930. addItems(itemParams, sel = null){
  931. itemParams.forEach(
  932. (elm, i) => {
  933. let value, text;
  934. if( typeof(elm) == "string" ){
  935. value = text = elm;
  936. }else{
  937. value = elm.value;
  938. text = elm.text || value;
  939. }
  940. const $item = document.createElement("option");
  941. $item.value = value || "option"+i;
  942. $item.textContent = text || $item.value;
  943. this.$combo.appendChild($item);
  944. }
  945. );
  946. if( sel != null ) this.selectItem(sel);
  947. }
  948. selectItem(target){
  949. if( typeof target == "number" ){
  950. this.$combo.selectedIndex = target;
  951. }else if( typeof target == "string" ){
  952. this.$combo.value = target;
  953. }
  954. }
  955. appendTo($to){
  956. $to.appendChild(this.$combo);
  957. }
  958. // isDefault(){
  959. // if( typeof this.defaultValue == "number" ){
  960. // return this.$combo.selectedIndex == this.defaultValue;
  961. // }else if( typeof this.defaultValue == "string" ){
  962. // return this.$combo.value == this.defaultValue;
  963. // }
  964. // return false;
  965. // }
  966. }
  967. // ラジオボタン管理クラス
  968. class RadioSet extends ConfigControl {
  969. constructor(name, options){
  970. super(name, options);
  971. this.selected_ = null;
  972. this.items = [];
  973.  
  974. if( options.items != null ){
  975. this.addItems( options.items, this.getInitial() );
  976. }
  977. }
  978. get state(){
  979. return this.selected_;
  980. }
  981. set state(val){
  982. if( val >= 0 && val < this.items.length ){
  983. this.selected_ = val;
  984. this.items[val].$radio.checked = true;
  985. this.onchange();
  986. }
  987. }
  988. selectedIndex(){
  989. return this.selected_;
  990. }
  991. selectedItem(){
  992. let i = this.selectedIndex();
  993. if( i < 0 || i >= this.items.length ){
  994. i = this.defaultValue;
  995. }
  996. return this.items[i];
  997. }
  998. // isDefault(){
  999. // return this.selectedIndex() == this.defaultValue;
  1000. // }
  1001.  
  1002. addItems(itemParams, sel = null){
  1003. itemParams.forEach(
  1004. (elm, i) => {
  1005. const nameItem = `${this.name}-item-${i}`;
  1006. const forId = elm.id || nameItem;
  1007. const value = elm.value || nameItem;
  1008. const title = elm.title || nameItem;
  1009. const text = elm.text || nameItem;
  1010.  
  1011. const $label = document.createElement("label");
  1012. $label.attr(
  1013. {
  1014. 'for': forId,
  1015. 'title': title,
  1016. }
  1017. );
  1018. const $radio = document.createElement("input");
  1019. $radio.attr(
  1020. {
  1021. 'type': "radio",
  1022. 'name': this.name,
  1023. 'id': forId,
  1024. }
  1025. );
  1026. $radio.addEventListener(
  1027. 'change', () => {
  1028. this.selected_ = i;
  1029. this.onchange();
  1030. }
  1031. );
  1032. $label.appendChild( $radio );
  1033. $label.appendChild( document.createTextNode(text) );
  1034.  
  1035. this.items.push(
  1036. {
  1037. 'value': value,
  1038. '$radio': $radio,
  1039. '$label': $label
  1040. }
  1041. );
  1042. }
  1043. );
  1044.  
  1045. if( sel != null ) this.state = sel;
  1046. }
  1047. appendTo($to){
  1048. this.items.forEach(
  1049. elm => $to.appendChild( elm.$label )
  1050. );
  1051. }
  1052. }
  1053.  
  1054. // コンフィグダイアログ
  1055. const $configDialog = document.createElement("div");
  1056. $configDialog.id = "apc-configDialog";
  1057. $configDialog.addEventListener(
  1058. 'click', () => config.close()
  1059. );
  1060. document.body.appendChild($configDialog);
  1061.  
  1062. /// コンフィグメニュー
  1063. const $config = document.createElement("div");
  1064. $config.id = "apc-config";
  1065. // バブリングストップ
  1066. $config.addEventListener(
  1067. 'click', e => e.stopPropagation()
  1068. );
  1069. $configDialog.appendChild($config);
  1070.  
  1071.  
  1072. const $createH4 = (title, text) => {
  1073. const $h4 = document.createElement("h4");
  1074. $h4.title = title;
  1075. $h4.textContent = text;
  1076. return $h4;
  1077. };
  1078. ///config/ 表示する条件
  1079. $config.appendChild( $createH4( "コンソールパネルが表示されるページ", "Page where APC-panel exists") );
  1080. const radioDisplayRule = new RadioSet(
  1081. "display_rule",
  1082. {
  1083. 'onchange': function(){
  1084. $panel.attr(this.name, this.selectedItem().value);
  1085. },
  1086. 'items': [
  1087. {
  1088. 'id': "apc-mdAlways",
  1089. 'value': "always",
  1090. 'text': "always",
  1091. 'title': "常に表示",
  1092. },
  1093. {
  1094. 'id': "apc-mdValid",
  1095. 'value': "valid",
  1096. 'text': "only when \"AutoPagerize\" is valid",
  1097. 'title': "AutoPagerizeが有効なページであれば表示\n(Alt+Ctrl+pで設定画面を表示)",
  1098. }
  1099. ]
  1100. }
  1101. );
  1102. // radioDisplayRule.isAlways = function(){
  1103. // return this.selectedItem() == 0;
  1104. // };
  1105. // radioDisplayRule.isValid = function(){
  1106. // return this.selectedItem() == 1;
  1107. // };
  1108. radioDisplayRule.appendTo($config);
  1109.  
  1110. ///config/position/ マウスが外れている時の表示
  1111. $config.appendChild(
  1112. $createH4(
  1113. "マウスが外れている時の表示",
  1114. "Appearance when non-hover"
  1115. )
  1116. );
  1117. const radioNonHover = new RadioSet(
  1118. "non_hover",
  1119. {
  1120. 'onchange': function(){
  1121. $panel.attr(this.name, this.selectedItem().value);
  1122. },
  1123. 'items': [
  1124. {
  1125. 'id': "apc-no_change",
  1126. 'value': "no_change",
  1127. 'text': "No change",
  1128. 'title': "変更しない",
  1129. },
  1130. {
  1131. 'id': "apc-transparent",
  1132. 'value': "transparent",
  1133. 'text': "Transparent",
  1134. 'title': "背景を透過",
  1135. },
  1136. {
  1137. 'id': "apc-stealth",
  1138. 'value': "stealth",
  1139. 'text': "Stealth",
  1140. 'title': "隠れる",
  1141. }
  1142. ]
  1143. }
  1144. );
  1145. radioNonHover.appendTo($config);
  1146.  
  1147. const $createInputRow = (title, text) => {
  1148. const $row = document.createElement("div");
  1149. $row.className = "apc-mItem";
  1150. $row.title = title;
  1151. $row.textContent = text;
  1152. return $row;
  1153. };
  1154. const $createGroupBox = subject => {
  1155. const $box = document.createElement("div");
  1156. $box.className = "apc-group";
  1157. const $subject = document.createElement("h5");
  1158. $subject.textContent = subject;
  1159. $box.appendChild($subject);
  1160. return $box;
  1161. };
  1162. /// 位置設定
  1163. const UpdatePanelPos = () => {
  1164. $panel.style[comboPanelPosY.value] =
  1165. numericPanelPosValue.value + comboPanelPosUnit.value;
  1166. };
  1167. const UpdateBoxPos = ($elm, val) => {
  1168. $elm.style.height = val+"px";
  1169. };
  1170. const UpdateAllPositon = () => {
  1171. UpdatePanelPos();
  1172. UpdateBoxPos( $spacingBox12, numericSpecing12.value);
  1173. UpdateBoxPos( $spacingBox23, numericSpecing23.value);
  1174. };
  1175. ///config/ 位置
  1176. $config.appendChild( $createH4( "位置設定", "Expression style") );
  1177.  
  1178. //config/position/ パネル位置
  1179. const $panelPos = $createInputRow(
  1180. "パネル位置",
  1181. "panel position Y:"
  1182. );
  1183. $config.appendChild($panelPos);
  1184.  
  1185. ///config/position/panelPos/ 使用単位
  1186. const comboPanelPosY = new ComboSet(
  1187. "panel_position_y", {
  1188. 'items': [
  1189. { 'value': "top", 'text': "Top"},
  1190. { 'value': "bottom", 'text': "Bottom"} ],
  1191. 'onchange': () => {
  1192. $panel.style.removeProperty("top");
  1193. $panel.style.removeProperty("bottom");
  1194. UpdatePanelPos();
  1195. }
  1196. }
  1197. );
  1198. comboPanelPosY.appendTo($panelPos);
  1199. ///config/position/panelPos/ 座標
  1200. const numericPanelPosValue = new NumericSet(
  1201. "panel_position_value", {
  1202. 'defaultValue': 48,
  1203. 'onchange': UpdatePanelPos,
  1204. }
  1205. );
  1206. numericPanelPosValue.appendTo($panelPos);
  1207. ///config/position/panelPos/ 使用単位
  1208. const comboPanelPosUnit = new ComboSet(
  1209. "panel_position_unit", {
  1210. 'items': ["px","%"],
  1211. 'onchange': UpdatePanelPos,
  1212. }
  1213. );
  1214. comboPanelPosUnit.appendTo($panelPos);
  1215. ///config/position/ オプションボックス
  1216. const $optionGroup = $createGroupBox("Option box");
  1217. $config.appendChild($optionGroup);
  1218. ///config/position/optionBox/ 並び順
  1219. const $optionOrder = $createInputRow( "並び順", "Order:");
  1220. $config.appendChild($optionOrder);
  1221. const numericOptionOrder = new NumericSet(
  1222. "option_order",
  1223. {
  1224. 'defaultValue': 1,
  1225. 'onchange': function(){
  1226. $optionBox.style.order = this.value*2-1;
  1227. },
  1228. },
  1229. { 'min': 1, 'max': 3}
  1230. );
  1231. numericOptionOrder.appendTo($optionOrder);
  1232. $optionGroup.appendChild($optionOrder);
  1233. /// 展開方向
  1234. const optionDirOptions = {
  1235. 'items':
  1236. [ "Left", "Right", "Up", "Down",
  1237. {value: "Up_protrude", text: "Up (protrude)"},
  1238. {value: "Down_protrude", text: "Down (protrude)"} ],
  1239. 'onchange': function(){
  1240. $optionBox.attr(this.name, this.value);
  1241. }
  1242. };
  1243. ///config/position/optionBox/ 有効ページでの展開方向
  1244. const $optionValidForm = $createInputRow(
  1245. "Autopagerizeが有効なページでの展開方向",
  1246. "Direction when AP is valid:"
  1247. );
  1248. const comboOptionDirValid = new ComboSet(
  1249. "option_dir_valid", optionDirOptions
  1250. );
  1251. comboOptionDirValid.appendTo($optionValidForm);
  1252. $optionGroup.appendChild($optionValidForm);
  1253. ///config/position/optionBox/ 対象外ページの展開方向
  1254. const $optionInvalidForm = $createInputRow(
  1255. "Autopagerizeが有効でないページでの展開方向",
  1256. "Direction when AP is invalid:"
  1257. );
  1258. const comboOptionDirInvalid = new ComboSet(
  1259. "option_dir_invalid", optionDirOptions
  1260. );
  1261. comboOptionDirInvalid.appendTo($optionInvalidForm);
  1262. $optionGroup.appendChild($optionInvalidForm);
  1263.  
  1264. ///config/position/scrollerBox/ BOX1とBOX2の間隔
  1265. const $specing12 = $createInputRow(
  1266. "BOX1とBOX2の間隔",
  1267. "Specing between to BOX1 and BOX2:"
  1268. );
  1269. const numericSpecing12 = new NumericSet(
  1270. "spacing_12",
  1271. {
  1272. 'onchange': function(){
  1273. UpdateBoxPos( $spacingBox12, this.value);
  1274. }
  1275. }
  1276. );
  1277. numericSpecing12.appendTo($specing12);
  1278. $specing12.appendChild(document.createTextNode(" px"));
  1279. $config.appendChild($specing12);
  1280.  
  1281. ///config/position/ スクローラーボックス
  1282. const $scrollerGroup = $createGroupBox("Scroller box");
  1283. $config.appendChild($scrollerGroup);
  1284. ///config/position/scrollerBox/ 並び順
  1285. const $scrollerOrder = $createInputRow( "並び順", "Order:");
  1286. $config.appendChild($scrollerOrder);
  1287. const numericScrollerOrder = new NumericSet(
  1288. "scroller_order",
  1289. {
  1290. 'defaultValue': 2,
  1291. 'onchange': function(){
  1292. $scrollerBox.style.order = this.value*2-1;
  1293. }
  1294. },
  1295. { 'min': 1, 'max': 3}
  1296. );
  1297. numericScrollerOrder.appendTo($scrollerOrder);
  1298. $scrollerGroup.appendChild($scrollerOrder);
  1299. ///config/position/scrollerBox/ 配置形状
  1300. const $scrollerForm = $createInputRow(
  1301. "配置形状\n(AP対象外ページでは自動的にボタンが2つになるので無視される)",
  1302. "Form when AP is valid:"
  1303. );
  1304. const comboScrollerForm = new ComboSet(
  1305. "scroller_form", {
  1306. 'items': ["Slim","Square"],
  1307. 'onchange': function(){
  1308. $scrollerBox.attr( this.name, this.value);
  1309. }
  1310. }
  1311. );
  1312. comboScrollerForm.appendTo($scrollerForm);
  1313. $scrollerGroup.appendChild($scrollerForm);
  1314.  
  1315. ///config/position/pageIndexBox/ BOX2とBOX3の間隔
  1316. const $specing23 = $createInputRow(
  1317. " BOX2とBOX3の間隔",
  1318. "Specing between to BOX2 and BOX3:"
  1319. );
  1320. const numericSpecing23 = new NumericSet(
  1321. "spacing_23",
  1322. {
  1323. 'onchange': function(){
  1324. UpdateBoxPos( $spacingBox23, this.value);
  1325. }
  1326. }
  1327. );
  1328. numericSpecing23.appendTo($specing23);
  1329. $specing23.appendChild(document.createTextNode(" px"));
  1330. $config.appendChild($specing23);
  1331. ///config/position/ ページインデックスボックス
  1332. const $pageIndexGroup = $createGroupBox("PageIndex box");
  1333. $config.appendChild($pageIndexGroup);
  1334. ///config/position/pageIndexBox/ 並び順
  1335. const $pageIndexOrder = $createInputRow( "並び順", "Order:");
  1336. $config.appendChild($pageIndexOrder);
  1337. const numericPageIndexOrder = new NumericSet(
  1338. "pageindex_order",
  1339. {
  1340. 'defaultValue': 3,
  1341. 'onchange': function(){
  1342. $pageIndexBox.style.order = this.value*2-1;
  1343. }
  1344. },
  1345. { 'min': 1, 'max': 3}
  1346. );
  1347. numericPageIndexOrder.appendTo($pageIndexOrder);
  1348. $pageIndexGroup.appendChild($pageIndexOrder);
  1349. ///config/position/pageIndexBox/ 配置形状
  1350. const $pageIndexForm = $createInputRow(
  1351. "配置形状\n(AP対象外ページでは自動的に非表示となるので無視される)",
  1352. "Form when AP is valid:"
  1353. );
  1354. const comboPageIndexForm = new ComboSet(
  1355. "page_index_form",
  1356. {
  1357. 'items': ["Pile", "Strip", "Growed Pile"],
  1358. 'onchange': function(){
  1359. $pageIndexBox.attr( this.name, this.value);
  1360. }
  1361. }
  1362. );
  1363. comboPageIndexForm.appendTo($pageIndexForm);
  1364. $pageIndexGroup.appendChild($pageIndexForm);
  1365. ///config/position/pageIndexBox/ 展開方向
  1366. const $pageListExpand = $createInputRow(
  1367. "展開方向",
  1368. "Page-list expand direction:"
  1369. );
  1370. const comboPageListExpand = new ComboSet(
  1371. "page_list_expand",
  1372. {
  1373. 'items': [
  1374. "Upper", "Lower",
  1375. "Left-upper", "Left-lower",
  1376. "Right-upper", "Right-lower"
  1377. ],
  1378. 'defaultValue': "Lower",
  1379. 'onchange': function(){
  1380. $pageIndexBox.attr( this.name, this.value);
  1381. }
  1382. }
  1383. );
  1384. comboPageListExpand.appendTo($pageListExpand);
  1385. $pageIndexGroup.appendChild($pageListExpand);
  1386.  
  1387.  
  1388.  
  1389. const $mButtons = document.createElement("div");
  1390. $mButtons.id = "apc-mButtons";
  1391. $config.appendChild($mButtons);
  1392.  
  1393. ///config/ 閉じるボタン
  1394. const $mCloseDecision = document.createElement("input");
  1395. $mCloseDecision.attr(
  1396. {
  1397. 'id': "apc-mCloseDecision",
  1398. 'class': "apc-config_button",
  1399. 'type': "button",
  1400. 'value': "OK",
  1401. }
  1402. );
  1403. $mCloseDecision.addEventListener(
  1404. 'click', () => config.close()
  1405. );
  1406. $mButtons.appendChild($mCloseDecision);
  1407. const $mCloseCancel = document.createElement("input");
  1408. $mCloseCancel.attr(
  1409. {
  1410. 'id': "apc-mCloseCancel",
  1411. 'class': "apc-config_button",
  1412. 'type': "button",
  1413. 'value': "Cancel",
  1414. }
  1415. );
  1416. $mCloseCancel.addEventListener(
  1417. 'click', () => config.close(false)
  1418. );
  1419. $mButtons.appendChild($mCloseCancel);
  1420.  
  1421.  
  1422. class Config {
  1423. constructor(name, options = null){
  1424. this.name = name;
  1425. this.tempParams = null;
  1426. this.init(options);
  1427. }
  1428. init(options){
  1429. if( options == null ) return;
  1430. if( options.items != null ) this.items = options.items;
  1431. if( options.onopen != null ) this.onopen_ = options.onopen;
  1432. if( options.onclose != null ) this.onclose_ = options.onclose;
  1433. }
  1434. open(){
  1435. this.tempParams = this.getParams();
  1436. if( typeof this.onopen_ == "function" ){ this.onopen_(); }
  1437. }
  1438. close(decision = true){
  1439. if( decision ) this.save();
  1440. else this.restore(this.tempParams);
  1441. this.tempParams = null;
  1442.  
  1443. if( typeof this.onclose_ == "function" ){ this.onclose_(); }
  1444. }
  1445. restore(params){
  1446. if( params == null ){
  1447. this.items.forEach( elm => elm.setDefault() );
  1448. }else{
  1449. this.items.forEach( elm => elm.restoreState(params) );
  1450. }
  1451. }
  1452. getParams(){
  1453. const params = {};
  1454. this.items.forEach(
  1455. elm => params[elm.name] = elm.state
  1456. );
  1457. return params;
  1458. }
  1459. save(){
  1460. GM.setValue( this.name, this.getParams() );
  1461. }
  1462. load(){
  1463. GM.getValue( this.name, null).then(
  1464. result => this.restore(result)
  1465. );
  1466. }
  1467. }
  1468. const config = new Config(
  1469. "config",
  1470. {
  1471. 'onopen': () => $panel.setAttribute("config", true),
  1472. 'onclose': () => {
  1473. $panel.removeAttribute("config");
  1474. UpdateAllPositon();
  1475. },
  1476. 'items': [
  1477. radioDisplayRule, radioNonHover,
  1478. comboPanelPosY,
  1479. numericPanelPosValue, comboPanelPosUnit,
  1480. numericOptionOrder,
  1481. comboOptionDirValid, comboOptionDirInvalid,
  1482. numericSpecing12,
  1483. numericScrollerOrder,
  1484. comboScrollerForm,
  1485. numericSpecing23,
  1486. numericPageIndexOrder,
  1487. comboPageIndexForm,
  1488. comboPageListExpand
  1489. ]
  1490. }
  1491. );
  1492. const loadState = () => {
  1493. GM.getValue( apEnable.name, true).then(
  1494. result => {
  1495. FireEvent(
  1496. result?
  1497. 'AutoPagerizeEnableRequest':
  1498. 'AutoPagerizeDisableRequest'
  1499. );
  1500. }
  1501. );
  1502. config.load();
  1503. UpdateAllPositon();
  1504. };
  1505. loadState();
  1506.  
  1507. // タブを切り替えた時にステータスを最新に同期する
  1508. window.document.addEventListener(
  1509. 'visibilitychange',
  1510. () => {
  1511. if( window.document.visibilityState == 'visible' ){
  1512. loadState();
  1513. }
  1514. }
  1515. );
  1516. })();