Questden-BLICK2

Improves readability of questden.org

// ==UserScript==
// @name         Questden-BLICK2
// @namespace    https://phi.pf-control.de/tgchan
// @version      2024-12-29
// @description  Improves readability of questden.org
// @author       dediggefedde
// @match *://questden.org/kusaba/draw/*
// @match *://questden.org/kusaba/meep/*
// @match *://questden.org/kusaba/moo/*
// @match *://questden.org/kusaba/quest/*
// @match *://questden.org/kusaba/questdis/*
// @match *://questden.org/kusaba/tg/*
// @match *://questden.org/kusaba/questarch/*
// @match *://questden.org/kusaba/graveyard/*
// @icon         
// @require      https://cdnjs.cloudflare.com/ajax/libs/jszip/3.6.0/jszip.min.js
// @grant        GM.xmlHttpRequest
// @grant        GM.setValue
// @grant        GM.getValue
// @license      MIT; http://opensource.org/licenses/MIT
// @sandbox      raw
// ==/UserScript==
// 
/* jshint esnext:true */
/* eslint curly: 0 */
//
//
//
(function () {
  'use strict';
  //
  const imgRes = {
    refreshimg: "%2F9hAAAABHNCSVQICAgIfAhkiAAAABl0RVh0U29mdHdhcmUAd3d3Lmlua3NjYXBlLm9yZ5vuPBoAAAMiSURBVDiNbVJNbFRlFD33e%2B%2BV%2Ba0zbQdbqLYKhhKmoaIzjX%2FRmEY3xGBYqLxEjJCB8LNhBxsMTQwRdGGi4AOjMUwlmDQBN%2FwEIZLUwqtx0dCS1NLGAUY7tB3ozLTz3vu%2B66LzkllwkpvcxT0nJ%2BdcYmb4SGWyzzTo2hEQ3vSkatWEmCfCuOPKEwAu2Jbp1t0eBHCCfIF0JrtF08RAT9cqfV1ns9HWFAGYUShWMDJ2v3xn%2BqEnFe%2B2LfNselc2w4zvAMSImZHKZNcbuhjZ2tcdiseC0InQHA0gHNARDRkIGgIzxUUczw5XCsWKHQ019D4uVw1mNBEz49U9Z0%2F1rGv7NBgwxK3RHKquB0PX5JrVscW%2B9PPh3vWtpAmCEIRfrk2ot15sFwe%2Bvuo4rmzRAYCZP7wzXRCup1gxLzLjM8eVZ8anZ5MT9%2BaPXu9o7tq3dVMoEQvh43e7RC0GAqAEAEipwlXH85IvPPuDlOoRgEHbMvO2ZV7549uPXhqdLOzdfewS7j8sYXqmDKkYAAMACyyvqumpSH9jJPBVZ%2FvKbgBTdWnruiZ2bnsn6XY%2BHUWicQU0Qb4D1mt38V8%2F37yAJ%2BO0J9VrF4cnq1ftqSXfvOOqAAD2W6gCMGoErpvfAOwE4D5BWLct854v4J0%2F%2Br4GEJgZF%2B1%2F1Knzf%2BWkVD22ZRbrWYezY6tK5aWO30fGbwCI6QBABBAR7v5XhiEYJwf%2FJABf1JNTmWyYiO72blx7Jj8zlwKgAagKABBEBAAdiRBWt4Tx5f63qa0lcuyVPT%2FfSmWy6ZrG9hWGrufyswfyheLrmkaubZmuWHZAeFxxcWFoij2P0b4yisM73gh90Jd8ubU5ci29a8A1dPGNVDI%2B92gBTY1B0oQYBAD%2Fkaj%2Fx6Glidzc5KXhyc7Mlk3hRCyI7rUJeq49HiqWXfw7W4LjeBQJGjh3ebTiSXUEAJYfSTFN5OZO3zy5LfmgsPBJ%2F%2Fc35o8P3CwN384jP1tG1ZUQRCjMl%2BS5y6OLUqmMbZljAOC38BOA7bZlci0wA8B7DYa2VzFvkFLFDV2bIcJQ1ZGHbMv82w%2F3fy5VdmoTZczGAAAAAElFTkSuQmCC",
    nextbutbild: "%2Ff4%2Bf%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2FyH5BAEKAA8ALAAAAAAQABAAAAQx8IlJa5U2awqQ2RVQIB4oiAdZEOZZtu4Hd3IrsnCu79nw%2BQKgBUAregIZonKJxMAeEQA7",
    wartbild: "%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2F%2FyH%2FC05FVFNDQVBFMi4wAwEAAAAh%2BQQJBQAfACwAAAAAMgAyAEAFk%2BAnjmRpnmiqrmzrigDwzq9M33Cc6ngb%2F7WfbSYE9o7IpHLJHA2bJ170CSUJV0ZmEVnseqXVsHhMLkPBZhR6nC1RyW23ebtbK7u%2BOA6%2F59f9VV9vaYSFhoeIiYqLLHaHg4iQgViEjh%2BSgY6YZ5qVek5pV2pzn06WSXR%2FmaJYpTeAja55rDSwKl5LgqZfYoK4jMBKIQAh%2BQQJBQAfACwAAAAAMgAyAEAFqOAnjmRpnmiqpgEArHBMtnJtf%2B6ty%2B67%2FytXYOi7CXvApHLJJBWbNSR0J5wCe0%2FmkHbFeltCYs5KLptFrcA5uI6N26oqPA5Qz1PYO15KfsN6dkpggUaAYChedUdlikRgfnqRkpM%2FfJKEknWUTptOmJNZlJCTRJ0ilqKhoKOZmqZ5r6xNrnS0U1syW6o%2FgzaPuzx1hzpHuG6Pxl2Aib%2BJtlpYv82fptU3IQAh%2BQQJBQAfACwAAAAAMgAyAEAFrOAnjmRpnmiqpkC7vvAYtEJs20Ee3Hzc5r1gLAfQ7YTIpHLJNM1cTRw0yiseqULiFIvUCYzW70%2FHLSt%2FpCuaOgOaVet36gmQr9pXO4pY08%2FbfndWgSo6LXWEe0RaRYknYlZajpOUN3g7X3qGeXtYkpSHnISfoHGTgJUfoalPoomklHipqlugOX2pPwWzqm6sVoi5q6yXs3iHwY50AqFFhzJfAbtlzqGLkbzZSiEAIfkECQUAHwAsAAAAADIAMgBABbvgJ45kaZ5oqp5BCwCtsJJFMLPtrZvHGry2nXD3%2BwGGyJ0gIGgmn9CodKpz%2Faii4AymxQ5bOS8UzBRjY4GCec0uCV5wuAtYxC6L6uehq5Ov%2BShAeW0%2BYIQ3SzCHM2SAiyZMRo8%2BXI6TJWAwMpecnZ6fkXJ3lmdzlShgm1NGMIpIpIxkVKpVp16usUttsFlAnyRgPb8fo8MirbyLmcbELsw1hsZWzEVX0olHxqPZ0mjMH3uR3x%2BpuuPn6G0hACH5BAkFAB8ALAAAAAAyADIAQAXO4CeOZGmeaKqeAxC8grB%2Bwgvc8%2FkGwpH%2FJJ4KJgMaSz5ULXBsOp%2FQqDS6i8GS0xGi5sp%2BYLMY15vCmrY1cqq46qnf8Li8tOQubfgbM%2Bu6Pl0CCEc8O28BZigBfWxxBSk7XXMfiEFgkiKMJTyUlycxA52hoqOkpaaXeDuphHI3fUQ%2BBwgFijaAWYQvnEOKAFB3e1AHYkcDqmQvP7Q8vobBa8dymSbRc890S46SB7t3pyPVktcjNWnfkwID03C7mDHn4O%2FwNPLz9vf4%2Bfr7oiEAIfkECQUAHwAsAAAAADIAMgBABdjgJ45kaZ5oqp6CELyBcASDitCugKxmTAS8YA8AFBqPpBhyGTwwn6MddEo9JgTE1ktQq4pyA1f1kAuGBV5VAOBMQwNtt3xOr09zMRjMteejqQN6cEs3e1YtLW4HRAA8ZH92i0UoYXYkDTqWR5OaKwNSnSkHoKEspaeoqap1BTFhezBEL3FpM1l5eS0Iu4%2BCVYh%2BkDyvW0x%2BAAW0SJLCQX5dVYY8CMBzN0Q8XMpyL9AnxZprKgMEobMpCNt1iasl7O0jAaTwg%2FAjze3qqvqp86v%2B9gIKHEiQSQgAIfkECQUAHwAsAAAAADIAMgBABdXgJ45kaZ5oqqJDIAwIUqDIICBCoOvH6v%2FA0S5ILOJuxaTqoFM6R4LZc0oFDmzYmwtR7dJygYG3qhOMnTnAmYjjrmm9t%2FzpEtjDtno5NyB4EwMLbkU5doZzRUw5iEFXdow%2FTGGQKwU6EJQqZZkokpwnBy%2BfJDijpqeoQQk4LntpLgVxZwdXfS8vLQUFhoV2V122YkGGO0hOMVV3LqkjrAHMUHayqa3CqQmF0B87g6gNm8yK3acITdABAM%2FnAQvnANOnNw3Qj8wBB%2BOmZqgx%2BZnW2gJSCQEAIfkECQUAHwAsAAAAADIAMgBABd7gJ45kaZ5oqq7syAgDDA9tbacOAht3bzuCgMBHROlgRSIiAAgEktBokYBASK%2BpgiAIsGKhW2fjW5SRazknLVr1ioKfrQDhOJzveMcisR0QCAMIfng%2BMUJzhDcxW3aJLToBAwmOLYtulChAQpgqYZecIwdOn6CiiFCkKgdBQpOgr7CxXwkEBgIHgJKyJQi5cq67A34DQmu7HwcFi0%2FHH0ebzR%2FEjNGLxsesQ9Vbqa9ASM2a2s1y0Uu3zQmL0U7jshHtDrsMq1vYo7tL7WQJ3SPPBRolanOrCgxg0RJGCQEAIfkECQUAHwAsAAAAADIAMgBABeTgJ45kaZ5oqq5s675wiQxIEt8tIggD7puCwMDmShwEvw9COOgln9CodDo4LKZPwoBHwN50gsOK6AUjng%2FHIcAOANq7YPjsrdvv3ofBgI85FgRkfStbTYMrCAU7TocoWwFijSc6A11jd4%2BRkiMKAwFIP5Q1LWtxAXR9CKibrK2ur7AnCQtXsSMLVauwCVq2IgQEuq6VW3yxBAY8tkafwqwHW86bTWy2BIuxCaIp0lSflq%2BPoIh2DnENRWXYMaNQcd0sCOg3CVs840laYDQJ2toMqpaw4cGozoEjQQaq2gHOl0McIQAAOw%3D%3D",
    stylebuts: "%2F7lV1dW5J0ijCRqN0ihYVrJkLByRjU2OywoWTDAsmHfJLD%2FDAmZZA2vDsot5CTYLeA1LWMAmGOM1wcYyDhK2ZcsKtrJGmhlN0KSezpXu74%2FqOD2hezRi7Xe7nmcelWrufOr0uVWn7j11%2B3wFE7Zt27apx44dU6LRqGKapqCCzePxyCNHjtiAk%2F2Zcqvxa%2Fz%2FzXwpZUXMiZsQQlbSbq75Ige94447lJt%2BucUa8XhJCA0pKjuPBMKOyTwrw%2FhjL14AjF999dXmDTfc4BQbvH37dvVj31lmDms1fo3%2Fv48PKJuvPmmdCX%2FXjzq0XADK3dBzzFcBWRwshJRS7Nq1S7vma63GmM%2BPpoKqQIV8pARHgm1DQyaN8vQrV3R2dvZv27YttW3bNgkoXV1d2utu8KZGvT5U1Y1KQsx8DimzP9T4Nf7zlw%2BoG9%2FZl54L%2Fu7bW32AXTRSOWO%2BI8F2oD6dZs%2F3WnXAFkK4geiJJ57wfODrC4yRSIi6gCDkE3g04QaJCgKE7YBpS9IGxNOSxngcvetvz7nwwgt7rr76arOurs7zpuuN2HAoRMgv8HlAUwVKhR%2FAkWDZkrQJ8ZRkXjyOfqLGr%2FGfH%2FwVK1Zol1%2FbE59L%2FgPfbg8BZm6GMxf8ZAZiaUnjeIyHblvkBUwhhNQ0TdONkI%2FWekFrvUJjWMGvg6q4J5npBJYjMUwYT0lOjzskNR%2BKoqyLx%2BMpj8cTVRQlIOt9LIwI5oUUQn6BVwNFqfADOJCxXOcMxx1Smg%2Blu8Z%2FzvI9Cbx9t6Ms%2FyDCTqB0fx9n6fue9%2F5ZPH6Al3b9mHuXvI3uyIqK%2BYCvmL%2B5M4xHo6ob2LRg15FYng%2BEgWS22Zz4J5pwGByHhPQBeAFHSmlrsVhMx6fRFFZY2KiwoE4h4BV4VDD7HihNxrReXnYSy4GMKRmNS3QNeqUHKcTqZDLZlU6nzWAwGPKGdJoiCk1hd4SiewSqIovTIEUzIlG0D7YjMExJUAdFURhwPEgpa%2FznKj99EL2pBbVOQPQINDVDRDkr9oueH01%2FA7RfPWf%2B2Tx%2BBL2piXVaN5nIqortB%2FzekCfPrw%2BCR4WMdRN2GlTf1Pb79U%2FiOGDa0Fwn8nwgWDTFCBbzQz7BDQt3uRECeJhLCsBLHi7s5vccPvtfm%2FGqAtuBVMYDoANpAG10dFQVqht1fB73R9dAU0BvuxBr6ImibG3pB3AcN18hHffvvBooqiCRTreNj49HLMvqTyaTmoMX6Ugyphs5DUsWjU7kJGmTonNIiWW7QUg6EgeBkUq1joyMRAzD6Kua3%2F19WPT2s8efC%2FsTQwhjAOltRPG2TM7PDCMzAzieRmxPy3PH%2FpFDGL55KEkJpw%2BBpx6S2WvZNmDo98h0v3t9CwVHa8Ke9zLSh29Btny4Ovtj8ekfkUk5J%2F0bsGJ4h7tJ2AYLEk8Ti1yeDxwz8QHFQeT5adO94dPGuItPl5ttpUHzQSL9SRzp4PV8qYSfzTvkk5PFfEUAhpEfPmRKyJlJ05TxtDuFsW2JKUsvAm10dFSaFiTSDrGkQFMcAl7hDlHQyYyM51HegFN2KTnZEcR4ShJNOMTTMB6NeoUQyokTJ8yFCxcmR%2BJ1SEsSjQs0NTv8qfD1S%2B4clg1JQzKaFqSHhnQhhOju7rYcx6mOr7wFeq2zxz8D%2B0ejDlrP%2F0U1R0Co7k2k%2BlAWvQd8C1yebSNPfB1hDCNRkNLGEV6che8hqTSfFftt08ZGJZGyGDO1afnq4T2oHe9AWBbywG7EoreBYYGVxOn6OiKyCXxbEIEOUIPI0d1Yh7%2BBlTxFWrersl8eOT79qzvNmpP%2Bvaj3PrYrHSyJ7qdzbDcp%2BQzdkVUV9e%2F69esTI3Enz6%2BPZG9ibiKWen3%2BHGYSPIHyfZAEfXBy0M7zgQRgZBuU8DUV8A%2FnuUMlHhma1E9H%2ByxMy72HYymRiyQOgNbX12clDUk0JulXIJVW0D3uyEARgmTP6Two4HXKni5Suv0fS0lG4w7DMcHo4KCVSCSsxx9%2FPD08PGwOxxaTSdhoqshnWAGs3dfkWdrG7015LJfJtWxJ3FFJDA5ayWTSnIlf6XamfGd4B073bW7eyLcE0ifytlfDFye%2BBuNPoix6J%2Br8rTiZfpwDn4KuG9HO%2FTf3IXzsa8jxJxHt1%2BTb2M9%2BEk7eiFz9lTn3j3XoC9lst8SyHdIrbpiUn3zoLWiam0IXXTcUAMducB94gWXg70ALXAFx3B9sYB2ObMA6eT3U21XZbx3rmtZ2LWKfef9KyRt3%2FpSvnv8NXnS6m9auw8xP3cP2VZ0V9e8rX%2FlKazgm8%2Fz5je6zP5pY6z7TY%2BAN50Y8xaMf93cLWvYRTboBIscHxoHcU84s5guAb308j%2Fk4byow31Q4XnSUE%2F1XZnMdknFLIRt8pBBCaidPnrTTFsQSDkhJMi1LkijjPYN5kM%2FZS2bwUZAO3qYteBrXu1lQC5IZSSzlMJ5WiUajdiKRMB999FHz6NGjZlx7FeQGR0Ud4LTfiLXXTWDpsWxnLvw85r4Plh6ThadZHIXY%2BLiTTqetmfj56HzwczixpxFSoL%2FgzkkDxJnwM3tvRm2%2FBq35pVijO3D6%2FjVvezV8R1mL9LXg8V4KMRvHCmDGPZA5hrfD5dnKuUhfC1pRGyPugUwXeoczK%2FunDRDqC7CH7kOZdyX22H0YaTk5f%2FX3EYDxxBvwvuDnkN3Xz8%2Ftb0Nb%2Bw6UIr8U%2FNMKnbchk9XZb4yOTWv7XFw%2Fa04%2FyjNOB33pAHvFcq4aHmZV%2FC7GWt%2BfbzgdH8jE0zLPH0%2FlzrEPgJTaSio5xQdQYTz7u2jMzvOzE5PcNMAq5rsDhaGqRhBjWT%2FZDqSFQvFrTu3JJ5%2B0jTWSREriOJBKZyNRNkCMDozmQcGQB9tZSbrvATj1K4LLdLTwSixbYliQykhStiSdTjtSSufZZ591Dh06JM3LwHQmuTiVZtKZ7Egp99JGbSs%2FVtQJpgKmaTqGYTh%2F%2BMMf7MWLFzMlP7ct%2B0esru9iRx8vYU4WJGbDz6Tj0P0L9PoXo4QuhA0%2FxzBnwY9sRfrHyOy%2FCSe2G6ykex0IgcjxIltxfGOkJ7SRgDCn5idPfA%2BsGE5sL0r4XJzxvYjAYmTyOCKyGWfsYbyrPocdewYn%2BjSke%2FF0fhLHsLEMC9WwsQwTy57ePxnDyNuazhh5f6czaXxq25z6P3%2BdTLXNwfVz0ZGf8psV78Uw4WBgLVY6TX36JAtP7%2BF4w4YZ%2BdkZWp5vWqX8kLcPgIb7FxEKqG6a7MLCyCjXPnc9mW7u0S5eKFXMFwJIFxIbpSmO9ORusgr22wrkgkNuiiHNVWA4DlIKMtnXI9kkKMNjscKcTlmM42lgZOzXblQ78Dsi5yxHZhdamJbElCqWZUkppezp6ZGAjAACOWlOTObDr5z2mJjsb4ATJ05My88%2FCUd3oETOK2GWLSmdBT%2F12Kuzvu8l8%2FjrSrPQF9xdFd%2BK7sE8%2BBkkCmr9C1CbX4F18nvI1NG83VO1IXU0b%2BNkfBk%2FhOMY6G1%2Fg1SD2LFD%2BFZcT%2BqJN%2BJb%2Bj7SBwaxRv6MGlyDmN%2BM1f0f2CM7wNOQ5cmK%2FCNzq28m2XfsDIqiz5n%2FpZQzzx3PgB%2FKDLO5%2Fz629N1bhj7%2F1G85Xr9%2BRr4QQkbeNlzwzxSmBv0Kp4bShP1avs3DdohL1HiWLyddFj2Rj4QvOU7RO4ovFU7ypcLxL03w01SDSs3NJbgZTByZjSCFBolkIfMZSjtIGSgcS3XhM5z8%2B1RHgiNACIGqqqKlpUWoqsoooEg56dNXZA%2BqpQ4oO1bc39Xw85txGu%2BCl5Qxz5QfuuDXOOkBUgf%2FEYx%2B9%2B99iwms%2B1rZ5TAT3zxyEwiV4IbbUfRI7lEBQuTtTk3TRsmeYDK%2BZ%2BHVONE%2FY574Fp72a1ByTKGiIlEVD0JKnOH7QVFR%2FAsRQqAgcbJtbaa2P%2FXYaxFCIoQg%2Ffhr8u3Sj78GpArCgxh7ErVxy5z5X8wwVzrT6%2BdFPb%2Fkl6s%2BwL2d780fu%2Br493nzM1%2Fk%2FIH7%2BcWa62bkd3d3i7q3j%2BT5AnjYDnOJGmMk0UBjcDT%2FN5GAh1BAZQwYSTQgfCLfPsThwlRmwvctivnuyFwpDNJLM%2FTT%2BmkyvlZYc50dzToTV0oWHbBdSu6YonrBzi7tlNk1nwpomqYEg0HlwgsvVL1er7xHcV%2BsTmqcooKQaJgIxYOUElVVssYZiAlPHFUBj8dTMR%2FAHPkzqqbii3TOOOeeDZ9AM97z%2Fp3MwG8x%2Bu9AGifxKNXxNcdBI4UItOH1RQo3gX0aRVXwKODM0EZXp%2BY7p76LFj4fEepATR%2FFtochtgdVyUD0CTC6UUgi7TRS9aFIAxnfDZ75KNZpiO9GtYewnOSk%2FIYX%2FgorupvU4S8QPv%2FOkn2A5LGvYp26Hf%2F88gCRGXoIo%2BvrhM%2F%2FWVX%2Bz10nU22T9UGlfOFYXHbqV3z9%2FFtLOE8ufDlvPPQVWjM9LI4f4lRk5bT8nJl60b2rSpUdsp6Lw2M8YtcTnhemQ9FJZCzCIZV9jYugMYQ6HGOHrEdV1LwN2Y%2BsTjxnsf2Kqhbd%2FErxLyYNFcWfr4gvAbTW1lYRU0AXElURZd%2FD8OqFxQ9ej4NjJvPHvPXL8emFNRFSgq5AIBBQGhsbtU2bNnl8Pp%2B8bx%2F4ppjjWZHFSKMPLfE4WmgN5ugOfF6v%2B7uDHyay%2FtaS6OZVwPH7K%2BYDOLGHCIQX4%2FfMPBqdDX%2F8yb%2FB2%2F4u6ttfhuH3kzrxnbJzzcQPKAqW7kUQxWP3g2OSOvEt%2FJoJmo5XtVA8GrbXB5O0kaoHXbHwavqk%2FIa1XwZrFKFuQ9E04P1uSmO%2Bm0SMLDi%2F4C9jHEWP4FhxFC1UYv%2BwosMU%2FklE78cfWYrfA4nxP%2Bb3AfyrPkLs0D9jHfoYwaUfQQkswjFGMQZ%2Fhxj8NZGlb8OrVed%2FM3udTLnQyDO7%2FvWZ47z28NcQXh9msL60L5UIvQvOY2l0H3%2FddQs%2FXvuPJLyNU%2FIB26sV%2BLtECz7NR6gxxK6RFt66uy2bUYVFre7nufJujWBA5c7zTUKNIeIjcXxagQ%2BUXF3FfCHg877C6qv%2Fc013fv%2BdV%2F45v3%2Feeefl9%2F%2Fzb%2FtL7M%2Fy3QCxadMm9VgafMJdPSmyQSI3rwr6C52gZY5hpocJ%2Br0IRaNp2SvQdFGyLDQhBFoopLS0tHjWrFmjq6oqvIcEfie7dFuUztf0Fe8m1f0fOIPfRzG30LDwamLRu%2FE0vBC98TJUbynfqwi0cLhivgQs2Y0SbsI58jHC535l2mWts%2BL7%2FWjOCcxn342CpLHzXfi8omq%2BuvxqjL6fw%2FFPuhc4CiLSjjSH8ZsH0CLrUJddTabvzpI2SqQd2xjCaxzAG9g8KT%2BoqShi%2FqT2T5xD463L%2Fhuuyn7bOo6n5QoC3tz%2B5Xk%2FCCC47npSPT%2FGOHkjjnQTZppWT2TNJ9AiG6v2v%2BX3TxsggkV9UE3%2F%2FsMT1zI%2FdQqAr%2B58Bf942d2k9HoAPvnQ22g0ByAQ4Lzkk5z32Ov49Nbfk1H9k%2FLdh2yBb2UXODgxh6v3toMX%2FvBig%2FV3SpobvPzhxQZX3e%2B2uXpvgF9uGiHgCRD0ijwf8BV3XTEfAcFgsNCVRUFUUZQp%2FVTsnyK%2B1JqamhS9V4Ii8GqybKFIMNRE%2FaKXYBlR4gO%2FRZoJ6ptWEWm7Et3fnLfScSCDwA84waBobGxUm5ubPZqmKX5VoqoCr5Al3%2B8QApi%2Flrr5pTdtcPOtE3NMOBIyUuBHIqvhA6augBgluOJ9ePRJc1hnxA9ekFvzcO1UObKK%2BPXtr4HWF2HF9oPw4qnbiJMZwIrtQQvOQ9Mh0PFqnNbLsGL7kcKLFtmAkxnEGH8aGZiPX63e%2FgpyfDPab4%2F%2BEd3no77txSgeSHtsd3%2FC2D2w7C2w7C1z4n%2FDX%2FkIohr%2B17b%2BtMQ%2FAsitW%2Fq%2FL76zzD8q4JuCD4hifkbzE2oIEh9N8NstaUL1Qfz4aW%2FRuXvjKCEtiMfr7gfrA%2FhG3SDo8xT4ue9K5O77ifb7s4FTCIGu6zMGCL%2Bn1D9FfKkFAgEZkDaGoqIICwWJUvQ8aVv7gayLJJGmCyY8eyQCgUSiCjDQCEqLjK4n6%2BvrrVAopACEhI0hVALScrOtYrL1%2F7LsewFuM5HPshqKRkjapD2eRDX8hrVfPav8ObVfi6A1XlSYE%2Fpb8fjbShKeqqcOtfHCwrPf34rH18qY8PyP2W8veAnzmq5ACAkC5mWnhme%2Ff6feFCGfC%2F0rivkAibFkPsEaH0sAcPfGRP7%2Fuf1cOyklKjLPzxqSTyhMtH%2Fpa3bkrdyZKNi%2F87Zi%2B%2Fvz149Cqf3FKQqtvr7ebrAy9GkhdMch4tjZbPRk6%2FPlpItxpYSE0BgXCovMFKai9Pj9%2FjFFUUwhhAhlMvToQYRUCDoWubc0w3vePWMnz1v%2FXZevaMSEQruRwlTV3kr4Vdlf4z%2Fv%2BOHVt06%2FyMuUs%2BLf%2BMDlVa3E%2FdTlf5ySD9jF%2FMj%2B%2B2a10jeqqHl%2B7nVB7g6fS%2F8vMlNkV2m6OYiGhgarvv89banW208NebwkEYSw0aWDUvzuuyiOFs9ZM0Iljkpa0ZhvpTl475avNTc3HwyHw6f9fn8KYEvDN5ZGT3%2F4%2BGmPj5gDQWnjkQ6hVd%2BekT9uCBIiyzfTHPrtlq%2FNnz%2F%2FQCX8Su2v8Wv8Yv4HL3mgOr41Ld954N%2Fb6zf9bd%2FYXNj%2FwG3t9RPv9Lnk%2F9Hl56OJBtiLFy%2BO9jxyVWfbpv8%2BMqJ66Vc8OBWXlIKQbdJixzl09%2FqbvV7vY7quP9vR0TGqqqoJ0NbWNqJvf%2F3KlpU%2FOzSieukX1fHDjkmLHePgXetu9vv9j%2Fn9%2FgM1fo3%2FPOFLgCdva%2FVf8PaTqTPhP%2FafHVNmZc8WX9u2bZsDZA4ePDhw%2FMFLl4%2BNjXXatr3ScZw2KWV44jvXSTZnXIjEKej2eDyHgMM%2Bn%2B%2F0%2BvXr04sWLZIA69evN%2B%2B%2F%2F%2F7%2B3Vm%2BaZorHMdZWCk%2FKkTilKLU%2BDX%2B85Kfe2g%2F9p8dHs58m2ql31nha0IIKaV09u%2Ffn%2Bzt7e0NhULReDx%2BTEoZkVJ6p1x%2BVQo0HMcZk1KOhsPh%2BNatW82tW7fmT7R161a2b9%2BeeOaZZwxgzHGco47jVMwXQhhAtMav8Z%2BP%2FOfzJib5fy5DKipwTnGQcIqSJ3Ka89X4Nf7%2FVv7zbtPcLKnMOUa599571e7ubnVsbExNJpMzTmSSyaQMBAIyEonYixYtEtu2bbOmcLwCiO3bt6uHDh1Savwav8avmm9n%2BfJs8EOhkHXttdda2TUUMj%2BCyJW%2B%2F8x%2FtBvepghqwIvQKgueUoIdS2EMjTP8xGtbOjo6xrdt22Zkcxv5qLx%2F%2F37147fOz%2Bjza%2Fwav8Z%2FLvJ7HnpJc0NDw%2FjWrVuNnK6HkFKK7du3q1%2B5Z6Ppa63PVpKqQheDwjLWdN8YR%2B65pLOtra3%2FoosuSv3TP%2F2TBJTDhw9r1323Ke1tqS%2BU06%2Bi5pnE%2FTp5pr%2FGr%2FFr%2FLnmO9mfVN8YT%2F90w3LHcfp6enrSgNTuuOMO5XsPXmZGOuvx6%2BDNl5urPEC4tSDAu6ieziv%2FeKR%2F52tW9vX19e7atctcsGCBdt0365KRznq8GnlhHncYM3N0I8u3bMjU%2BDV%2BjX9W%2BIYFWns961%2Fz2NE9v75geWdnZ%2B%2BRI0cMLRQKaYHmCEGfpM4PIZ9b1bqauv22AykDEhlQ2iP0SXluOp1O%2Bny%2BqKIo3rqFEcJ%2Bid%2FjBiBNqU75xy2tDykTxMIa%2F7nMV%2B0xjj36Oc696ivYRpTDD32Bc6788vPeP82ndrHl0S%2Bz8%2BJPMNh63qz5Pl2gqtn7q5LMZ1a1Lm3Is%2Bcf27U9nganNYJt26sSicQ4MKINDQ1p3rAXvyYJeyWRrDqPqsCBnd8vAa5%2B4Tsmyhq4xS5t0LOSX3ZER0q5KhaLHU8kEmld1%2F2BSCt%2BTeL3yOwIRRTSw9PLMripYUe67aXAjujYtr2yxn9u8mN9O6iLzMOvSYa6dxIJN%2BDT5Fmx%2F%2BCfrp%2F2Blh12RfmzD%2FL%2Bx5GDQToGN7N%2BKKNs7bfp6uoCiQy78RKghaY2v6Q73Y3R6CCkE7F9n9auzsfHe7itQXga%2B8q7BZ1%2BhfueDVIMFWBHvLiOM5K0zSPAGPa6OioqqgKQlpIKXAciZ2t7bB43es58lTh22u2IyeNQLadrQdhS1A0Uq4uQHhgYEDoui5tFAzDco2wBIqQVWt%2FWrbEMCU2GvF4vPX06dOh2fAPbf80K7f%2B81njz4X9RqybZLQLX7iVYMPySbnp8V4SY8fxhlrx1y9%2Fztjf2%2FUEvnAL4ylJ9%2FEn8AabGU9mRWaMJD17%2FoPk2DGktECo%2BCIdtK9%2FL3vv%2B%2F9YecU3q7J%2FbGx8WtvHU3JO%2BteXGsbTe5hkxqTh4IOMr3gbuMVdq%2Bbb2aF%2FPDXqniNVbreRAD0I48nXIZGEfHdhZCq3H086P8eYvial2y%2FxlFtT1jQlUtHIZDJttm2HAUXr6emR1kJIZyRJzRXBKVS1DnF6OJrHjcXllEOUjOlqc6YdGBsb81mWpR49etRKp9MiZWzEsCWaAooyu1fEjuMO5WzV5TuOMyt%2B%2FYbPMzjqnDX%2BmdivSJODD16HmRgAoSBxULUAKy%2B5AV%2Bk3fW5bfLs9usw4oU2iuqn8%2BLP4Qm1nxX7HcsExYNhmEivNi3%2F%2BNG9rHjhKxkcdTh6eA%2BdWz7B4KiDnYnxzPbrmNfxIgKtbyI0bzWaN8xo90PsuOczJKIDNCdkVfZ3n%2Bqf1u55Rf18Jv277ulf8lToAtqij9HeuxMOPMVgy3mzun5UX25Q8iuiyc2FoBADPVy0X1RwN22CnZaV268XAme0xMropH4ajsp8HsJUIJ1OBx3H8QJC6%2Bvrc%2BxWSFsSBYlhiJIk5amiqtaLouXOc7L1KA0LUoYko0AsFpPpdNp59NFHrSVLltip5m2QKdTky0Xnx%2B54WZ5zwbbfTnkspzsgJeB1%2BYZhzMivdDtT%2FlDX%2FRx7%2FGYkJqH6FSTGjuRtr4Z%2F9JEvMNr3DEs2f5jm5a8gOd7Lvt9fy9i9n2TDK%2F8TgMMP%2FzOjfc%2ByeNOH8m32%2Fu49jP72E6x%2FxQ%2Fn3D%2F77%2F8oIHAcBwmc%2B4pbJuU%2F9oOX5AuWDN1TqNV4%2Bp7rQCgEG1YRbDiX4KK%2FASBhAqZEabiE%2BhULOfn7vyOelFXZf3p4ZFrbo3F55v0rJQsf%2Bym%2FfP1PMQYGaR76Dc1P%2FZrDl26c1fWjOu45RhOLAciMgzfirmJIJLLA7H5mHFpaTxBNgp0dDVViP%2F%2FyibxtnyiuUXJt4fi1JX66Ji8AJHUwTTO%2FNko7cuSI07YBpOmWvc8YMi%2F8KRAMjxai0eFnH6PrmXuRjsWic15K69KLioZYbpCwPZBKpWQmk3GOHTvmdHd3y%2BbXg53V9Cr2%2FYorbuPpe9ziIamM%2B%2FvOy%2F%2BdPf%2F99pJjRVNWVA0ymYw0DMPZsWOH3d3dzVT83PbsA39PdPBJBApb3vTHKRNCs%2BXvf%2FiLLN74fppXvJahngc5%2FchnS2yvlO9reiHzAsuItL%2BcVEZiiQiWCJMYOpnneZu20OhfWtLGViKMDvWQMeWs7J9uC7Vewelj9zJ%2F6cs5ffy%2F3dHiJPwNr88G%2BJ9uZcubtgPw5%2BL9n11By%2FqPlvkFQASWsP5191ZtfzKVmtb2ubh%2BOk5upydyDlER4WT9Ws5PJOjYdxepF1xfEkkq5WvZ0nF%2B%2FYRro28%2Bqamq9%2Fvc0QOAZcjK7c9HGleCiyn%2Blx%2BhGK5sAm6GAMuykNnS3FpfX59sdsAxHBzbHWLkRg9CQDRWUPUwnCDhlgs4vu%2FXDD%2FyI0wnxLzWtfk3GbYNUgHbtqXjOHLXrl0OIF%2F9OnCs8mG9x9eCZWVFO7K%2F1%2F3tZcdKLibp8gHZ09Mje3p6puTntpWX%2FitdT36D0b4dkzLPlG8aKbqf%2BQmNi19OQ8ulnP%2BGP055nun4jYuuwkqP8cz2TzE%2B%2BCS2kSCni5HjNbZP3caxnCn5XbtuxTLGGRt8irr564me3k2gbinJ6FHqFmxhpHc752z9MuOD%2BxgfeIJ0rJuVF30ey3SwbBvbcrBtJ5tVn9o%2FlmXnbS3ZN028gYVz6v%2FcdTLVNhfXz5rdP%2BCxF3wI23I4NW8jtmURinYzv3cXAy2bqubbE05TF3DFbBruay%2FoYlx0oijvl5vmOZXbbxUWc5Yu67Rm9JOgtGy%2Fll9NZUkcG2wxddl7f91KVE8TieQdABx48i7Ov%2FLc%2FChCSvcDOI4j5UTRAmeGue9kv3eqmC%2FP0Hbk1MPULTi%2FOmYF%2FJ13bHWHivEeHrvjxSW%2Fe%2BG27VXxo4NP8eyfPgZSoaHtQpo7X0f3nn8nMXYof%2F5K2ky2xYcPYNsZOs55B6oeJjZygNUX38jjv3gFy8%2F%2Fe4zkAKM9jxJpXIcv0MLJp29luOchdF9jIdk0nQ6FI2fct4w0muadU%2F%2FP2d9M0taXHKLz%2BO9YdfQ3Zb9bcfg3DCw4b85sCQVU%2BoYzhP35QvMluhjV3D9fKtaYKdHFKGpToW3aiRMn5HrAsd1SYU4uQGRfSZlFUdqyJGih%2FLHR4ZNYdmF4Iil8u0XTtLzuAIB0ijLDRfy8toEs18VwI075wo6JugbT8XP%2FGokBmpe9svwiP0P%2BC%2F%2F6AdKJPp7903UYSbfQqS%2B8lA0vua1q%2Bw89%2Bo8IVDa%2F%2Bk40bwQEdOf8kf3jfJtX3Ynmy7bJ%2BlFOoYsB0L7mnYz1P8Lx3V%2BlY93%2FQeQ%2BhOJWOFdUtzDh4Il7EYqGL7LI%2Fby5PiqpZVnO33nnFQhcXYw%2F33l5vp27ryKEh%2FGBx2lsu3jO%2FC8qedF%2FBvy1B3%2FGoxd8jMc3fzA%2Fh9j09HfY%2BugNrOj6HQ9fdH3V%2FNnqYryQ%2Fsrtr1IXgwm6GMV8bfHixe51kP2tUlx2ZkInCGT%2BInCDgBeRrceXE1%2BaTHcgJkAtLoNVxHc%2FoATHQJmgiyHtDKpaWvVUEdXxAUZ6H7II4FsAABWzSURBVEHTNCINKymX9uKM%2BYFgC5tf%2FgMGj91N7%2BEfYyS6CtV8KuQLaSOcJMFIO7o3nI%2B4ljGIqiqugI3jFNr4itpk3DYqckp%2B995v0tCyhVDdUtLRQ9jGEOODTyJIEz21g0y8C6wEtpVC0fxIxyA%2B%2BAR6YAFmeoD4wONYmUEcMz4p%2F6I33Ed0cBcHd3yaC17725J9gCNPfJGefd9ifttFZf4f6nmA47u%2BzAte%2B99V%2BX8mXQwl99SaRf8Kx2LtwZ%2Fxq5ffnhckAjjU%2BWoue%2BwmGuMnWTDyLEONq6viz1YXQym6P2eyv1pdjOLPl%2BPrui7OOeccRWttbRWKAjJbEXfiCi%2BvrhWdzMa2Evljjc0r0bKGyexDTlPLdTEecUAoctLln5GGpRjJXhKDjxJuXMdo30N5XYyDD76bDVd9vyS6aRr4J%2BgaTMcHSazvAeoaluBR5RSrgs6Mv%2Bs3r2HRuX%2FHwhWvwucL0PX01%2FGoTnV8TeD1%2BhD2GFbqFI5jcnL3V%2FEIA49XRxUmHo%2BGz%2BfPtunFcax8G033oGDimYK%2F4YqvY6ZHWHzO36BoOpz%2FUQCa%2FupeQLCgvSBoY6WjaL4IlpFE04Ol9gdCU%2FpnrOe31DUsx6NKoj2%2Fy%2B67fjhnyyc4uOMzHHro71h23j%2Fgr1uEkR5h8NhvGDj2S5au%2F1s0RVblf98MuhjF%2FV1N%2F%2BqZcS5%2B4l8RPj9WsKGE4wTqGGrfTOvpPVy6%2B2buu%2FifSQfmVcyfrS6GpsiK7a9WF2PHZ2Jl9jc0NGgXXXSRR9u0aZM6oICiSjQBiiryOn9CQChQOFk6eohU%2FDShgA9F9bBm%2FavxakXOkwJVg9AEXYzHj7jZV4FEKKJkmL9q8%2Fs4ue9WBg99B7P1QtpXX0O0%2B%2Bc0tF7CvIWX51fhua%2BS3GWq4Qm6BtPxEQIn00U4soAjj%2Fwda190y6TDhzPhB%2Fx%2BnNRRnr3%2FzSAkKze%2FNyt0Uh1%2F%2Bfp30H%2F4p3T9%2BcNZtkJdYwdm6jTW%2BD4i8zewfN3b6Tv8U7r%2B%2FJGSNkZyEHN8H6HmSyfn6woBfd6k9k%2Bc4hGK5KJByfBzJvud5BFal1yFT5PYqSO0LrkSn6fA33Dp5%2Bh59vucfOp6HNtdtKPqDZx70fXUNW1GQlX%2BD8ygi%2BHTZte%2Fb77nrdTHewH4wC%2B38h9vuJ%2BMz9UKufruvyKS7IdAgBUjj7Pi7pfw7Tf%2BCUsLVMQPzFIXw6fJiu2vVhfDp8ky%2FyxcuFBftWqVV2tqalJOSwfFo6BJu7BOPNuvdXVNLFv9KtLpMXqO3IWVSdKycA2LV7yMUKQ1f105EncROA7BCboD2lEH1asgbKvkCSCApuZ1NDXfUmLwC176vbLRuZQSqaoIUR0fJD5dQTgjrN78Ebza5LOLM%2BG%2F8JU%2FzR75wJSzl0r4HStfT9vSK4gN7UWoXuoXbMJIDBAdfpJwaD5eDRav%2Bqt8G1QvdU2bMJIDjJ3eRSiyAG0W9jPtzLQy%2B4f7f4%2FX56N92UvxaKBpFouWvXTCqA2Wr3sHy9e9Y078759BF0PXZsf%2F%2BRvuKnvlmTvTndt%2BU%2F5qMztMr4Tvm6Uuhq5Vbn%2B1uhi6Nrl%2FlixZommBQEA6yQyaT4dk0l1crhQeKJsvLSx6aV10YekF5BQpFktQvB7sdAZ9gi6GTGdQfR6cmIUsWsZaaY45n98IunzPBF2DmfhrL7u15EKZa%2F5c2q9pdTS2XlK4MINtLAi2ldiueepoKG4TaKO5ow0l7P8fs7911atpbrsy956MjZffPqW%2F58r%2FlYGfW%2F0Ls9PFwKnc%2Fsve9VTett4iO%2F%2FtrqL%2F3BWd0n5d15ORSMQMh8OKVl9fbxv7xvEsa8a2LJxUBiGqW1AjARHwIT066VOnUSboYqRPx%2FAtXoCtW8hEJt8BTz%2Fw1hnZGy7%2FoZuMDnpRPTrpU4OoE3QNpuJXs5Kyxn%2F%2B8c%2B5%2BPZpuabhzIr%2Fvjsvqsr%2BW%2F760Yr5a%2FofnpV%2FrKDnL%2BR%2F9%2F4NBoNRVVUtraGhwTIOvaNNCfzklLe5HilUZDKDtKyCuNfEV4dF4VXoGkrAi%2BLzkj4VZcePN5bpYixTv7L08Kl%2FOK4vqMeWKk4ijbRsVl34gxn5GUdBCfpQfV7Sp8bY%2BePzynQNpuJXZr9a49f4JfyvvmFndfzA%2F0v%2BKb1%2F87oYj%2Fzmks7FW%2B8%2Fos%2BPoIZDhaoTFYwejFgK89QAj%2Fxw9ZS6GNt%2F%2FLKV886%2F65A%2BP4ISCSNUpeIIZ8ZSJHoHePgHK6fUNajxa%2Fwaf274xfdviS7GvrsvmJUuhhAiQSW6GFl%2BtboDQoiEUomuQY1f49f4c8qv6WLU%2BDV%2BjT8lv6aLUePX%2BDX%2BlPyaLkaNX%2BPX%2BDldDLK6GGJSXYxP%2FVu9oSiiooKXJSvUskvex468Zdq6%2FR%2B7yZ%2FJFeSo8Wv8Gv85xJeSvn1%2FNbUuxhe%2Fs8hUVYGmiep1MZxsyS1bcvKp10xat%2F%2FDn1PTquoWBM19S7DSE0jp1sO0bWr8Gr%2FGn2O%2BW%2BNDYtnw7EOXl%2BhiiJ%2F97Gfqbb%2FaZHm9Cn6fwKsrbpCocPYiHdd405RkDEkmIxk88OaVGzZs6L322mvNBQsWaO%2F%2FjJXUdYFXdwOQqrrLQCsp%2By2l6xjLcvmGUePX%2BDX%2BXPMNU5LOOGQykv1%2Funy5rusFXQzdoxAOKtRFVMJBBa%2Bu5MvOzTQ8yY0cUmmHeMIhlnCQE%2Br2BwKCUEDF7xf4vAqaKqqs2%2B8an0pJ4km7xn8O8zUlQW%2Fvt%2Bjs%2FDi2HePkiW%2BzbPnHnvf%2BaRvbx8XHvstDy6%2Blv27NrPn157bPShdjbH%2FPWfOPaeXuX8E4drkuhscj8PsVImGVurCCT1dQNTh2srSSzrKOV5W%2FRHXcctle3TXKsmWZLobP68XvEwT8WbZa3RzJtrMjGulg2UqZLkCN%2F9zhJ%2BN7iYQa8PsEYyP7CIfq8HnFWbG%2Fq%2Btr09q%2BZMmH5sw%2FK5JPoQYCLMkcYMx77qzt9%2BvMShfD8CsV239D586qdDE%2B8%2BUtgIJpSnRNlOti5L97IQtTBizBwgWX03XqvjzKtqeavxSqWyMp08WwbB3DlIi0xLJsd%2FhTYQTNJVBMyx0GWbYs0wWoht91%2FKssWfqRs8afC%2FvNzCDpdA%2B6bwEB%2F6JJ2Zn0AKlUDx69CZ9%2F0XPG%2Fr6Bp%2FH5mhiPO%2FT270HX5xGLZ%2Bsx2mn6%2B39OOtWDxAJUdG8rLS1v5ODBL9Gx%2BDNV2T%2BjLkbcmZP%2B9Rlj6ANHSVgG8%2BM7iTW8DrJFEarla5nZ6WKk0rJi%2B0lXqYuRcDBMiWm55e%2FLdTGy85tEykFKiccjcN9m%2BEp0MeY32JMMUVznZDIOiaRDKiPLdTFSL8CybLSkQ%2B4tSTWbO5VxnWOak%2BgCVMH3hz%2FAwJB11vhnYr8SMzne9SVscxiEQCJRVR9LOj6E19%2Fijthsi6PH%2FgXLGMq3URQvHe0fQtWbz4r9tuWK3BiGhe2o0%2FJPnDzA4o7LGByyONb1DB2L3sPAkIVlJjl67IvU11%2BA3%2FcSAoGlaJ4QY2NP8PgTN5NKD1LXaFdl%2F0y6GKF6a07697wT%2F80T6lrax56mY%2FQpRPBp%2BuvXzur6ycRmp4uRGLUqtp%2Fx6nQxhkYsVxfDkGQMp1wXw1LcOYgQuLoYqphUF2NBY%2Fkr2NwIwjAdUmlJKuWU6WIkU5tRMm65LUEhNO%2Fd87d5zrr1t015zK0c5pbld5xyXYCp%2BJXfwWfGHx3eQU%2Fv7UgsAr7FpNInC7ZXwe8%2B8Q3GY4doX%2Fh2GudvJZ3p5%2FChzzA69mVWrf4XAE4c%2FybjsUMsbHtbvs2hg9czMvovrFz15Tn3z9EjN%2BYfBI4jWbb805Py9z%2F2LvfiFDA0clP%2B708P3wQoBAJL8fmWo%2Ftejg24xdJtEOcRCM2n59RnicWdquyfSRejJWqfef9KyaIDd%2FGTzd%2FENE%2FTOvQHWg7%2FnmdXnjOr68dKzk4XIxGzK7f%2FE1XqYoxflbffsmS5LkbdcldZS0qHjJHNgrqrJUp0MY6e3ENP%2F0NIx2Fh80U0zduYF9ywLJmNQLJMFyO0%2BL15p4ui2nkdS27kwLNu6bN0xh2GLlr8OQ4euK7kWC5K5zpjoi7AVPzCRf5FYvF9CCnYcN4PpnzKz5Z%2F6Oi3WNj6VuYveDGjo48xPPq1Etsr5eu%2B9dSJdgLhy0hnHCwriGkFSCR68jzdt44IC0vaWHaQROIUhilnZf90m8%2B3hZHRP9FQfxkjow%2B62fRJ%2BKvWuAFxz9NvZcPGHwLw9O7i%2FbezcvU1ZX4BEOpCVq7%2BbtX2z6SLMRfXz9KhnZz0dBJ1gnTpK9iSSLC06%2Ffc2%2FGBUl2MCvladmRQrS5GOiMrt79qXQwnXxNCMokuRmiJxMy%2B8lBNgVAKAaJYFyOT8RPwraW77wFGonezcqmf%2BsiKfN7Cstwk5URdjCs6Jl8cqmlNBQ2MbLD3eFrLjpUNeSfoAkzFzyerlv4DPd0%2FIDr%2B5JTMM%2BFbZppT%2Fb%2BhruFFRCIXsHbdD6c9z1T8uvpLsYxxjhz6CrH4fhw7SV4XI8uL1F2K5Z%2B8jeNMze%2Ft%2BS9sK04svp9waDWx%2BDP4fIvIpE8SDG0gGt1JZ%2BeniMcPEYvtwUj3sWTpR7Fsd17rvoefXBej2D%2BWZedtLd030T0tc%2Br%2FGXUx5uD6WXviFzy67F3YNnSH12JbFmHrFAtG9tNXNM2olD9rXQynCvur1cWwJz4sJ9HFcN8%2BgCWKV2LJEl0Mn3cxggYSSbdS8eHj97N%2BdWdhCCpz87FJdDH%2Bh7fo%2BOOEQuvnnLv7qTcDYBr97Hn6bSW%2F23jef1XFisX2c%2FTIjYCgLrKJxqar6O%2F9Mal0V1VtJttSiaM40qC15Q2oWpBE4hjLl1%2FHnt3vZPHid3P0yBDR6C5CgVXoehN9vT8kOvYEmqd%2BznxlOQaaovN82XyZEVYNPsiagfvLfrd64P4pA8Rsthl1MarYqtbFmGbTTpw4IZdvKYpSE0p6lehi2BKhBvLHxmKnsB1ZMjwpVMct1cXIJlXLh5fZg0JMf6x4GDeZLsBU%2FPznMIZparpi2jaz4Z%2B36b%2FIpAc5cuRGTHMQAK93Eees%2BVLV%2FK7jNyOEwtpzb0XTQwAMnBL5rPpMbYr9N5Hf0rqN8dguentvp63tLYVMvaIiBCiKB4CR0T%2BBouL1txXaCFHWPxP5u5%2B6GiFcSYSnd78l3%2B7p3W8BqSCERiK2l%2Fr6zXPm%2F5l0Mc70%2BtnYdw9%2FWvF37Fx2Tf7YC7p%2BxJUHb2bV6e1sX%2F3hqvmz1cU4T%2Byu3P4qdTEm1i0u18XI5WVEQXZvsk4orPPO6mKoev5YvoCnWl63v9%2Ba%2BuZVFdWV48Is08UAA2XCE0fK6vgA0dEn0DSVcHhZRW8cquX7AwtYt%2F6rDA3ez8DpX2EaPVOuRJ2KL6WDIE3A34LuCxU9dYddXQwFVxdjmjaqOjW%2Ff%2BAHRMLnEQwuIpPuwnFGicf3IRSD%2BPhTGEYvgjS2nUKofpAGieRedM88LHuIZGIPtjOMdJKT8jef%2FyNi0b0cO%2F6vbNj4%2FZJ9gBNdt9Df9yMaG8sDxOjIDrpPfpv1G2%2Bvyv8z6mIos%2B9f4Vhs6LubOzbdXMJ5tu2lXHH0m8zLnKI5cYTT4c6q%2BLPWxVAqt79qXQyl3P4yXQxFcVW9J67AKtbF0DQwrXT%2BWH3dMjzZsubFymwTdTH%2B8KSYMkqHwx2YZj%2Fp1FMEA6sZjz6W1zs4dvSTrFnzldIkEOW6ANPxAZKJnUTCHegeMePTazb8vXveRWvr22hbeCX%2BgJ%2Be7u%2BXnWtmvopX96KIGNI5jSNNek%2Fehq5ZoOloqo3i0fB5fYhJ2kjVg6raKEKblL9mzeexrTE09XUoqga8x311Pc9N2s5vKuhMWkYMTQ9jWUm07EqeSvwTi%2F%2BJSHgxukcQTzyU3wdYseL9HD36bxw%2Fdj1LOt6P19%2BGYYwyMvRHTg%2FdS8fiv8Gjiar8P5MuRnEfVNO%2FXjPOpUdvRfh8WMEGdK2Io9RxumkDbePPcPnJ2%2FjdOR8n5W2smD9bXQxPkQ0z2V%2BtLsaD7zhWZn%2BJLsaxqMCjuZoWatF74Im6GIZ5gnRmxNXFUDRWL7sKn1cpCRCOlGW6GA%2FuFdlhbPndtaLz7fT0%2FpCRoZ9g12%2Bmre2NxGO%2FI1J3Pg0NF%2Bf57hTInc5M1AWYju%2FOw3oJh%2BZzout6Vq26ccoLarb8gN%2BPpJujhz%2Forjhd9tYSuyvlL136JgYH7qLn5A1Z%2FyvURdoxzWGkdQRfeA1Llr6Rwf7yNoYxhG0cxhdaMwVfRfHPr2zi6a3L%2Fhuqyn5pn6S5%2BVJ8XgVpn8jv57Zz11xH36k7OdX7rzjSzW2pWh2rV32UcGRd1f6fURdjQh9Uyn%2FHUx%2BkPt0HwN8%2F%2Fgb%2B%2FeKfkc7mYq559N3UWYMQCLAqtZtVT76FWy69C1P1V8SfrS6G36dUbH%2FVuhhZPxXzS3QxTsRd4Q1dE6jZb3PmXptEQvNZvPAKMkaUvtP3Y1tJmuevoL11KwF%2FU35u4cjCtzon1u13F165UxNFlM59fN5zmTfvptK538aby6J%2BIQFaHR%2FAp6sgxuhc%2Fp6SZb9zxd%2B8%2BVu5GD3lqKESftvCV9DSegnx2LOgeKkLr8PMnCYW30coOA%2BvV7Co%2FRW0tl5CPHYAKXQi2TbR8T0EA%2FPweETV9lcyqprJ%2FtHRP%2BH1eWlr3YrHI9A8DgvbLs%2BPMPOvDZdug6Xb5sT%2FM%2BlieHUxK%2F5PLvvP8tFK9t%2BfXP6jMv8ogKdC%2Fmx1MTweUbH91epieHUxqX%2Fyuhi5%2FALF4qXZbcOawpKK5qZJEkw5IVVc%2BT0EZboYbo6jdGRSyVai4Zs9gRCyTBdgJv7qc754Vvlzab%2BmRmhoKMjgef0t%2BVWU%2BameGqG%2B%2FoJCx%2FtaaPK1uAFa%2Fs%2FYP3%2Fei2hqelE%2BJ7X23Jv%2FYv37fOPPRhejGv5Fr3wgb%2BOJotXoX%2Fp2sfUnyj9Lll%2Bmi%2BF%2B3xyE5Q4zlCq%2FT577J%2FeudqIuhm2DjSzLbzyz%2F%2F0zPMEk5669pSzHMVEXYCq%2BqMz0Gv95yl%2Bx4qvT8g1Tzor%2Fwe0vq%2BgtSe5N%2Fte3%2FrZi%2Fsq998zKP6aUfzH%2Fl%2Bli2IPva2PBLaccR5a90qp0y8F3%2Ff6Skrr6iqKIzsZblj7T%2F97jEy1e1vmNGbn5FWTZT7TrD5Xzq%2F8QNX6ND1%2B%2B8N7quEWrHP%2Bf8w9ZXYwTT76m0zTdb4wZxix%2BTMmDv9h4s23bO4rq6meEEOm2traRwQNvXmnk%2BLP8efCXNX6NX%2BP%2FJfl5XYydO3f2777%2F4mXxeLzTcZyVs6mrryjKQeDIZHX7f%2FGLX%2FTtvu%2BiZfF4vNOyrJWO47QBFfGBhKIoPZqm1fg1fo3%2FF%2BRr2eq19mc%2F%2B9nUj370o1OpVCpqWdax2dTV1zRtyrr9W7duTfb29ub5QNV8RVFq%2FBq%2Fxv8L8mu6GDV%2BjV%2FjT8n%2F%2FwEPV3J1%2Fv5S1QAAAABJRU5ErkJggg%3D%3D"
  };
  //
  function rgb2hsl(r, g, b) {
    r /= 255;
    g /= 255;
    b /= 255;
    const max = Math.max(r, g, b);
    const min = Math.min(r, g, b);
    const delta = max - min;
    //
    const lum = (max + min) / 2;
    let hue = 0, sat = 0;
    //
    if (delta !== 0) {
      sat = lum > 0.5 ? delta / (2 - max - min) : delta / (max + min);
      if (max == r)
        hue = ((g - b) / delta + (g < b ? 6 : 0)) * 60;
      else if (max == g)
        hue = ((b - r) / delta + 2) * 60;
      else if (max == b)
        hue = ((r - g) / delta + 4) * 60;
    }
    return [Math.round(hue), Math.round(sat * 100), Math.round(lum * 100)];
  }
  ;
  //
  class Storage {
    //
    constructor(name, defaults) {
      this.name = name;
      this.default = defaults;
      this.data = { ...defaults }; // Initiale Kopie von defaults in data
    }
    reset() {
      this.data = { ...this.default };
      this.save();
    }
    save() {
      return GM.setValue(this.name, this.stringify());
    }
    async load() {
      const storedData = await GM.getValue(this.name, null);
      this.read(storedData);
    }
    read(str) {
      if (str === null) {
        this.data = { ...this.default };
        return;
      }
      try {
        const parsedData = this.parse(str);
        this.data = { ...this.default };
        for (const key of Object.keys(parsedData)) {
          this.data[key] = parsedData[key];
        }
      }
      catch (ex) {
        console.error(`BLICK error: Failed to parse stored settings for ${this.name}:`, ex);
        this.data = { ...this.default };
      }
    }
    set(key, value = this.default[key]) {
      if (key in this.data) {
        this.data[key] = value ?? this.default[key];
        this.save();
      }
      else {
        console.warn(`Blick error: Property ${String(key)} does not exist on Settings`);
      }
    }
    static replacer(key, value) {
      if (value instanceof Map) {
        return { _type: 'Map', value: Array.from(value.entries()) };
      }
      return value;
    }
    static reviver(key, value) {
      if (value && value._type === 'Map') {
        return new Map(value.value);
      }
      return value;
    }
    stringify() {
      return JSON.stringify(this.data, Storage.replacer);
    }
    parse(str) {
      return JSON.parse(str, Storage.reviver);
    }
  }
  class LoginCredents extends Storage {
    constructor() {
      let def = {
        name: "",
        pw: "",
        savelogin: true,
        token: "",
        exp: null
      };
      super("loginCredents", def);
    }
  }
  class Sync {
    constructor() {
      this.access = 0;
      this.serverURL = "https://phi.pf-control.de/tgchan/API.php";
      this.creds = new LoginCredents();
    }
    async login(username, password, savelogin) {
      this.creds.data.savelogin = savelogin;
      if (savelogin) {
        this.creds.data.name = username;
        this.creds.data.pw = password;
      }
      else {
        this.creds.data.name = this.creds.data.pw = "";
      }
      this.creds.save();
      return new Promise((resolve, reject) => {
        GM.xmlHttpRequest({
          method: 'POST',
          url: this.serverURL,
          headers: {
            'Content-Type': 'application/json',
          },
          data: JSON.stringify({
            type: "test",
            username: username,
            password: password
          }),
          onload: (response) => {
            if (response.status === 200) {
              try {
                const result = JSON.parse(response.responseText);
                if (result.token !== undefined) {
                  this.creds.data.token = result.token;
                  this.access = (result.access !== undefined) ? parseInt(result.access) : 0;
                  this.creds.data.name = result.name ?? "undefined";
                  this.creds.data.exp = new Date();
                  this.creds.save();
                  resolve(true);
                }
                else {
                  this.creds.data.token = "";
                  this.creds.data.exp = null;
                  this.creds.save();
                  resolve(false);
                }
              }
              catch (ex) {
                console.log("BLICK login error:", ex, response.responseText);
                reject(`Error: ${response.status}, ${response.response}`);
              }
            }
            else {
              this.creds.data.token = "";
              this.creds.data.exp = null;
              this.creds.save();
              try {
                const result = JSON.parse(response.responseText);
                reject(`Error: ${response.status}, ${result.error}`);
              }
              catch (ex) {
                reject(`Error: ${response.status}, ${response.response}`);
              }
            }
          },
          onerror: () => {
            this.creds.data.token = "";
            this.creds.data.exp = null;
            this.creds.save();
            reject('Request failed');
          }
        });
      });
    }
    async upload(name, storage) {
      if (storage === null)
        return;
      if (!this.creds.data.token) {
        throw new Error('Not authenticated');
      }
      return new Promise((resolve, reject) => {
        GM.xmlHttpRequest({
          method: 'POST',
          url: this.serverURL,
          headers: {
            'Content-Type': 'application/json',
            'Authorization': `Bearer ${this.creds.data.token}`
          },
          data: JSON.stringify({
            type: 'upload',
            token: this.creds.data.token,
            data: storage.data,
            obj: name
          }, Storage.replacer),
          onload: (response) => {
            if (response.status === 200) {
              try {
                let result = JSON.parse(response.responseText);
                if (result.success === undefined) {
                  reject(`Error: ${response.status}, ${result}`);
                }
                else {
                  if (result.access !== undefined)
                    this.access = result.access;
                  resolve();
                }
              }
              catch (ex) {
                console.log("BLICK upload error ", ex);
                reject(`Error: ${response.status}, ${response.response}`);
              }
            }
            else {
              try {
                const result = JSON.parse(response.responseText);
                reject(`Error: ${response.status}, ${result.error}`);
              }
              catch (ex) {
                console.log("BLICK upload error ", ex);
                reject(`Error: ${response.status}, ${response.response}`);
              }
            }
          },
          onerror: () => {
            reject('Request failed');
          }
        });
      });
    }
    async download(name, storage) {
      if (storage === null)
        return;
      if (!this.creds.data.token) {
        throw new Error('Not authenticated');
      }
      return new Promise((resolve, reject) => {
        GM.xmlHttpRequest({
          method: 'POST',
          url: this.serverURL,
          data: JSON.stringify({
            type: 'download',
            token: this.creds.data.token,
            obj: name
          }),
          headers: {
            'Authorization': `Bearer ${this.creds.data.token}`
          },
          onload: (response) => {
            if (response.status === 200) {
              // const serverData: T = JSON.parse(response.responseText);
              storage.read(response.responseText);
              // storage.data = { ...serverData };
              resolve();
            }
            else {
              try {
                const result = JSON.parse(response.responseText);
                reject(`Error: ${response.status}, ${result.error}`);
              }
              catch (ex) {
                reject(`Error: ${response.status}, ${response.response}`);
              }
            }
          },
          onerror: () => {
            reject('Request failed');
          }
        });
      });
    }
  }
  class Settings extends Storage {
    //
    constructor() {
      const def = {
        changeFont: true,
        fontsize: 12,
        fonttype: 1,
        paragraphMargin: true,
        imageHover: true,
        invertCol: false,
        replyForm: true,
        manageWatchlist: true,
        autoUpdWbar: false
      };
      super("settings", def);
      this.fontTypeList = ["unset", "Georgia", "Palatino Linotype", "Book Antiqua", "Tahoma", "Arial", "Helvetica", "Verdana", "Times New Roman"];
    }
    ;
  }
  class EditorConfig extends Storage {
    constructor() {
      let def = {
        colorList: Array(15).fill("#ffffff"),
        iconList: [
          { label: "Favourite", arr: [] },
          { label: "+", arr: [] }
        ],
        saveText: "",
        autoSave: true
      };
      super("editorConfig", def);
    }
  }
  //
  //GUI interaction
  class Sidebar {
    constructor(dom, set, syncr) {
      this.barStyle = `
      #BLICK_bar {width:20px;overflow:hidden auto;position:fixed;transition:width 0.5s,opacity 0.5s, height 0.5s; right:0px;height:30px;background-color:#9ad;z-index:9;opacity:0.5;top:50px;border:2px ridge #ddf;border-top-left-radius:25px;border-bottom-left-radius:25px;}
      #BLICK_bar .BLICK_cont {width:250px;font: 16px/2em georgia, Palatino Linotype, Book Antiqua, Tahoma;padding-left: 20px;visibility:hidden;}
      #BLICK_bar:hover {width:270px;opacity:1;height:min(450px, calc(95vh - 50px));}
      #BLICK_bar:hover .BLICK_cont{visibility:visible;}
      #BLICK_bar .BLICK_button {background-color: #DDDDFF;border: 1px ridge #CCCCCC;border-radius: 10px 10px 10px 10px;cursor: pointer;display: inline-block;font: 30px/17px georgia;height: 20px;margin: 5px;text-align: center;vertical-align: middle;width: 20px;}
      #BLICK_bar .BLICK_button:hover {background-color: #aaf;}
      #BLICK_bar .BLICK_button:active {background-color: #77d;}
      #BLICK_bar input {margin: 5px;}
      #BLICK_bar label #BLICK_bar input[type=checkbox] {cursor:pointer;}
      #BLICK_bar input[type="submit"]:hover {opacity: 0.9;background-color: #32AFFB;}
      #BLICK_bar input[type="submit"] {cursor: pointer;padding: 4px 8px;font-size: 14px;font-family: Arial, sans-serif;border: none;border-radius: 5px;transition: background-color 0.5s;background-color: #2196F3;color: white;font-weight:bold;}
      #BLICK_bar select {cursor:pointer;font-size: 16px;}
      #BLICK_bar .BLICK_Sec {background: #ccf;padding: 0 5px;border-radius: 5px;width: 230px;  margin: 5px 0;}
      #BLICK_bar .BLICK_sec_buts {display:flex;flex-direction: row;justify-content: space-evenly;}
      #BLICK_imgbox {position:fixed;top:50%;left:50%;transform:translate(-50%,-50%);max-width:90vw;max-height:90vh;z-index:777;display:none;}
      #BLICK_imgbox .loading{position:absolute;top:50%;left:50%;transform:translate(-50%,-50%)}
      #BLICK_imgbox .imgEl{width:100%;height:auto;display:none;}
      #BLICK_sync_updownSec{display:none;flex-direction: row;justify-content: space-evenly;}
      #BLICK_sync_loginHello{color:green;font-weight:bold;white-space:nowrap;overflow: hidden;}
      #BLICK_sync_loginAccess{padding-left:10px;color:green;font-size: 8pt;height: 10px;line-height: 3px;}
      #BLICK_fonttype{width:130px;}
      .BLICK_floatButs{border:none;background:none;cursor:pointer;position:absolute;top:0;width:16px;height:16px;padding:0;}
      .BLICK_floatButs:hover{filter:brightness(120%);}
      #BLICK_set_export{right:0px}
      #BLICK_set_import{right:18px}
      #BLICK_bar #BLICK_fontsize{width: 30px;height: 20px;margin: 0;box-sizing: content-box;padding: 0;vertical-align: middle;text-align: center;border: 1px solid #DDDDF0; z-index: 2;position: relative;}
      #BLICK_bar #BLICK_plus{border-radius: 10px 0px 0px 10px;margin:0;}
      #BLICK_bar #BLICK_minus{border-radius: 0px 10px 10px 0px;margin:0;}
  `;
      this.barHTML = `
      <div class='BLICK_cont' >
        <div class="BLICK_row">
          <button id='BLICK_set_export' class='BLICK_floatButs' title='Export settings to file'>💾</button>
          <button id='BLICK_set_import' class='BLICK_floatButs' title='Import settings from file'>📁</button>
          <input id='BLICK_set_importFile' type="file" id="fileInput" style="display: none;" accept=".json">
          <input type="checkbox" id="BLICK_changeFont" checked="ckecked">
          <label class="BLICK_title">Font size:</label>
          <span class="BLICK_button" id="BLICK_plus">+</span>
          <input type="text" id="BLICK_fontsize" value="0"/>
          <span class="BLICK_button" id="BLICK_minus">-</span>
          <br/>
          <label style='margin-left: 28px;' class="BLICK_title">Font type: </label>
          <select id="BLICK_fonttype">
          </select>
        </div>
        <div class="BLICK_row">
          <span class="BLICK_title">
            <input type="checkbox" id="BLICK_paragraphMargin" alt="Paragraph margin" checked="ckecked">
            <label for="BLICK_paragraphMargin">Paragraph margin</label>
          </span>
        </div>
        <div class="BLICK_row">
          <span class="BLICK_title">
            <input type="checkbox" id="BLICK_imageHover" alt="View image by hover" checked="ckecked">
            <label for="BLICK_imageHover">View image by hover</label>
          </span>
        </div>
        <div class="BLICK_row">
          <span class="BLICK_title">
            <input type="checkbox" id="BLICK_invertCol" alt="Invert colours">
            <label for="BLICK_invertCol">Invert colours</label>
          </span>
        </div>
        <div class="BLICK_row">
          <span class="BLICK_title">
            <input type="checkbox" id="BLICK_replyForm" alt="Change reply form" checked="ckecked">
            <label for="BLICK_replyForm">Change reply form</label>
          </span>
        </div>
        <div class="BLICK_row">
          <span class="BLICK_title">
            <input type="checkbox" id="BLICK_manageWatchlist" alt="Manage watch list" checked="ckecked">
            <label for="BLICK_manageWatchlist">Manage watch list</label>
          </span>
        </div>
        <div class="BLICK_row BLICK_Sec">
          <span class="BLICK_title">Synchronize Webserver:</span>
          <br>
          <div id='BLICK_sync_loginSec'>
            <input type="input" placeholder="name" value="" id="BLICK_sync_user" style="width:30%;" />
            <input style="width:30%;" type="password" placeholder="password" value="" id="BLICK_sync_pass" />
            <label for="BLICK_sync_savelogin" style='width:70%;display: block;cursor: pointer;'><input type="checkbox" id='BLICK_sync_savelogin'/>Store Login Data</label>
            <input type="submit" onclick="return false;" value="Login" id="BLICK_sync_loggin" />
            <a href="https://phi.pf-control.de/tgchan/reg.php" target="_blank">Register Account</a>
          </div>
          <div id='BLICK_sync_loginHello'></div>
          <div id='BLICK_sync_loginAccess'></div>
          <div id='BLICK_sync_updownSec'>
            <input type='submit' onclick='return false;' id='BLICK_sync_upload' value='Upload'/>
            <input type='submit' onclick='return false;' id='BLICK_sync_download' value='Download'/>
          </div>
        </div>
        <div class="BLICK_row BLICK_Sec">
          <span class="BLICK_title">Export Thread to file:</span>
          <br>
          <div class='BLICK_read_readExpButs'>
            <input type="submit" id="BLICK_epub" value="Epub" onclick="return false;">
            <input type="submit" id="BLICK_cbz" value="CBZ" onclick="return false;">
          </div>
        </div>
      </div>
  `;
      this.dom = dom;
      this.syncr = syncr;
      this.setting = set;
      this.addElements();
      this.addEvents();
      this.fexp = new FileExport();
    }
    restart() {
      this.loadSettings();
      this.dom.watchbar?.destroy();
      this.dom.watchbar = null;
      this.dom.repform.removeBar();
      this.dom.initStyles(); //resets watchbar and repform
    }
    //
    loadSettings() {
      let el;
      for (const key in this.setting.data) {
        el = document.querySelector("#BLICK_" + key);
        if (el === null)
          continue;
        if (typeof (this.setting.data[key]) === "boolean")
          el.checked = this.setting.data[key];
        if (typeof (this.setting.data[key]) === "number")
          el.value = String(this.setting.data[key]);
      }
      el = document.querySelector(`#BLICK_fonttype option[value='${this.setting.data.fonttype}']`);
      if (el !== null)
        el.setAttribute("selected", "selected");
      this.syncr.creds.load().then(() => {
        document.getElementById("BLICK_sync_user")?.setAttribute("value", this.syncr.creds.data.name);
        document.getElementById("BLICK_sync_pass")?.setAttribute("value", this.syncr.creds.data.pw);
        const checkbox = document.getElementById("BLICK_sync_savelogin");
        if (checkbox) {
          checkbox.checked = this.syncr.creds.data.savelogin;
        }
      });
    }
    //
    fillFontList() {
      const cont = document.querySelector("#BLICK_fonttype");
      if (cont === null)
        return console.log("BLICK error: language selector not found");
      cont.innerHTML = this.setting.fontTypeList.map((lang, index) => {
        return `<option value='${index}'>${lang}</option>`;
      }).join('');
    }
    //
    addElements() {
      this.tabSet = document.createElement("div");
      this.tabSet.id = "BLICK_bar";
      this.tabSet.innerHTML = this.barHTML;
      document.body.appendChild(this.tabSet);
      this.fillFontList();
      //
      this.tabStyle = document.createElement("style");
      this.tabStyle.id = "BLICK_barStyle";
      this.tabStyle.innerHTML = this.barStyle;
      document.body.appendChild(this.tabStyle);
    }
    //
    addEvents() {
      document.getElementById("BLICK_plus")?.addEventListener("click", () => {
        let num = this.setting.data.fontsize + 1;
        if (num <= 6)
          num = 6;
        this.setting.set("fontsize", num);
        document.getElementById("BLICK_fontsize").value = String(this.setting.data.fontsize);
        this.dom.setFontsize(true);
      });
      document.getElementById("BLICK_minus")?.addEventListener("click", () => {
        let num = this.setting.data.fontsize - 1;
        if (num < 6)
          return;
        this.setting.set("fontsize", num);
        document.getElementById("BLICK_fontsize").value = String(this.setting.data.fontsize);
        this.dom.setFontsize(true);
      });
      document.getElementById("BLICK_fontsize")?.addEventListener("exit", (ev) => {
        let num = parseInt(ev.target?.value ?? this.setting.default.fontsize);
        if (num > 6)
          this.setting.set("fontsize", num);
        else
          ev.target.value = String(this.setting.data.fontsize);
        this.dom.setFontsize(true);
      });
      document.getElementById("BLICK_changeFont")?.addEventListener("change", (ev) => {
        if (!(ev.target instanceof HTMLInputElement))
          return;
        const state = ev.target?.checked;
        this.setting.set("changeFont", state);
        this.dom.setFontsize(true);
      });
      document.getElementById("BLICK_fonttype")?.addEventListener("change", (ev) => {
        if (!(ev.target instanceof HTMLSelectElement))
          return;
        const sel = parseInt(ev.target?.value ?? 0);
        if (sel >= 0 && sel < this.setting.fontTypeList.length)
          this.setting.set("fonttype", sel);
        else
          this.setting.set("fonttype");
        this.dom.setFontsize(true);
      });
      document.getElementById("BLICK_paragraphMargin")?.addEventListener("change", (ev) => {
        if (!(ev.target instanceof HTMLInputElement))
          return;
        this.setting.set("paragraphMargin", ev.target?.checked);
        this.dom.setMargin(true);
      });
      document.getElementById("BLICK_invertCol")?.addEventListener("change", (ev) => {
        if (!(ev.target instanceof HTMLInputElement))
          return;
        this.setting.set("invertCol", ev.target?.checked);
        this.dom.setInvert(true);
      });
      document.getElementById("BLICK_imageHover")?.addEventListener("change", (ev) => {
        if (!(ev.target instanceof HTMLInputElement))
          return;
        this.setting.set("imageHover", ev.target?.checked);
        this.dom.setHoverImg();
      });
      document.getElementById("BLICK_replyForm")?.addEventListener("change", (ev) => {
        if (!(ev.target instanceof HTMLInputElement))
          return;
        this.setting.set("replyForm", ev.target?.checked);
        this.dom.setReplyForm(true);
      });
      document.getElementById("BLICK_manageWatchlist")?.addEventListener("click", (ev) => {
        if (!(ev.target instanceof HTMLInputElement))
          return;
        this.setting.set("manageWatchlist", ev.target.checked);
        this.dom.setWatchbar();
      });
      document.getElementById("BLICK_sync_loggin")?.addEventListener("click", (ev) => {
        let nam = document.getElementById("BLICK_sync_user")?.value;
        let pw = document.getElementById("BLICK_sync_pass")?.value;
        let savelogin = document.getElementById("BLICK_sync_savelogin")?.checked;
        this.syncr.login(nam, pw, savelogin).then((ret) => {
          if (ret === true) { //logged in
            document.getElementById("BLICK_sync_updownSec")?.style.setProperty("display", "flex");
            document.getElementById("BLICK_sync_loginSec")?.style.setProperty("display", "none");
            let loginHello = document.getElementById("BLICK_sync_loginHello");
            if (loginHello)
              loginHello.innerHTML = "Hello, " + this.syncr.creds.data.name + "!";
            //
            let loginAccess = document.getElementById("BLICK_sync_loginAccess");
            if (!loginAccess)
              return;
            if (this.syncr.access > 0) {
              loginAccess.innerHTML = "Last uploaded: " + this.formatDate(this.syncr.access * 1000);
            }
            else {
              loginAccess.innerHTML = "No data uploaded";
            }
          }
          else {
            alert(`Login failed without reason!`);
          }
        }).catch(ex => {
          alert(`Login failed!\n${ex}`);
          console.log("Login failed", ex);
        });
      });
      document.getElementById("BLICK_sync_upload")?.addEventListener("click", (ev) => {
        let loginHello = document.getElementById("BLICK_sync_loginHello");
        let loginAccess = document.getElementById("BLICK_sync_loginAccess");
        if (loginHello)
          loginHello.innerHTML = "Connecting...";
        Promise.all([
          this.syncr.upload("sidebar", this.setting),
          this.syncr.upload("editor", this.dom.repform.editConf),
          this.syncr.upload("watchbar", this.dom.watchbar?.wdata ?? null)
        ]).then(res => {
          if (loginHello)
            loginHello.innerHTML = "Upload successful!";
          if (loginAccess)
            loginAccess.innerHTML = "Last uploaded: " + this.formatDate(this.syncr.access * 1000);
          console.log("Upload successful!");
        }).catch(ex => {
          if (loginHello)
            loginHello.innerHTML = "Error.";
          alert(`Upload failed!\n${ex}`);
          console.log("Upload failed", ex);
        });
      });
      document.getElementById("BLICK_sync_download")?.addEventListener("click", (ev) => {
        let loginHello = document.getElementById("BLICK_sync_loginHello");
        if (loginHello)
          loginHello.innerHTML = "Connecting...";
        Promise.all([
          this.syncr.download("sidebar", this.setting),
          this.syncr.download("editor", this.dom.repform.editConf),
          this.syncr.download("watchbar", this.dom.watchbar?.wdata ?? null),
        ]).then(res => {
          Promise.all([
            this.setting.save(),
            this.dom.repform.editConf.save(),
            this.dom.watchbar?.wdata.save()
          ]).then(() => {
            if (loginHello)
              loginHello.innerHTML = "Download successful!";
            console.log("Download successful!");
            // alert("Download successful!\nRefresh the page to update the settings.");
            this.restart();
            //						alert("Data imported! Reload the page to apply the settings.");
          });
        }).catch(ex => {
          if (loginHello)
            loginHello.innerHTML = "Error.";
          alert(`Download failed!\n${ex}`);
          console.log("Download failed", ex);
        });
      });
      document.getElementById("BLICK_set_export")?.addEventListener("click", () => {
        let expObj = {
          settings: this.setting.stringify(),
          editor: this.dom.repform.editConf.stringify(),
          watchbar: this.dom.watchbar?.wdata.stringify() ?? null
        };
        const jsonString = JSON.stringify(expObj, null, 2);
        const blob = new Blob([jsonString], { type: 'application/json' });
        const url = URL.createObjectURL(blob);
        const a = document.createElement('a');
        a.href = url;
        a.download = 'questden_BLICK_settings.json';
        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);
        URL.revokeObjectURL(url);
      });
      document.getElementById("BLICK_set_import")?.addEventListener("click", () => {
        document.getElementById('BLICK_set_importFile')?.click();
      });
      document.getElementById("BLICK_set_importFile")?.addEventListener("change", (event) => {
        if (event.target == null)
          return;
        const file = event.target.files?.item(0) ?? null;
        if (file === null)
          return;
        const reader = new FileReader();
        reader.onload = (e) => {
          try {
            const jsonString = e.target?.result || null;
            if (jsonString === null)
              return;
            const jsonData = JSON.parse(jsonString);
            if (jsonData.settings !== undefined) {
              this.setting.read(jsonData.settings);
              this.setting.save();
            }
            if (jsonData.editor !== undefined) {
              this.dom.repform.editConf.read(jsonData.editor);
              this.dom.repform.editConf.save();
            }
            if (jsonData.watchbar !== undefined) {
              this.dom.watchbar?.wdata.read(jsonData.watchbar);
              this.dom.watchbar?.wdata.save();
            }
            this.restart();
            //						alert("Data imported! Reload the page to apply the settings.");
          }
          catch (error) {
            console.error('Fehler beim Parsen der JSON-Datei:', error);
          }
        };
        reader.readAsText(file); // Lese die Datei als Text
      });
      document.getElementById("BLICK_epub")?.addEventListener("click", () => {
        this.fexp.showExportForm("epub");
      });
      document.getElementById("BLICK_cbz")?.addEventListener("click", () => {
        this.fexp.showExportForm("cbz");
      });
    }
    formatDate(uixTS = Date.now()) {
      const date = new Date(uixTS);
      const userTimezone = Intl.DateTimeFormat().resolvedOptions().timeZone;
      const options = {
        year: 'numeric', month: '2-digit', day: '2-digit',
        hour: '2-digit', minute: '2-digit', second: '2-digit',
        timeZone: userTimezone
      };
      return new Intl.DateTimeFormat('de-DE', options).format(date);
    }
  }
  ;
  //
  //
  class ImageManager {
    constructor(container, conf, clickCallback) {
      this.clickCallback = clickCallback;
      this.container = container;
      this.editConf = conf;
      this.selectedItems = [];
      this.startX = 0;
      this.startY = 0;
      this.selectionBox = null;
      this.allIcons = [];
      this.shownIcons = [];
      this.curActTab = -1;
      this.spliterDraggin = false;
      if (this.editConf.data.iconList === null) {
        this.editConf.set("iconList", [
          { label: "Favourite", arr: [] },
          { label: "+", arr: [] }
        ]);
      }
      this.itemCount = 0;
      this.itemsPerLoad = 40;
      this.initialize();
    }
    //
    initialize() {
      this.insertStyle();
      this.insertHTML();
      this.setupEventListeners();
      this.loadTabs();
      this.loadIcons();
    }
    //
    insertHTML() {
      this.pickerDiv = document.createElement("div");
      this.pickerDiv.innerHTML = `
          <div id="BLICK_iP_tabColumn">
            <input id="BLICK_iP_seachInp" type="text" placeholder="Search...">
            <div id="BLICK_iP_tabs"></div>
          </div>
          <div id="BLICK_iP_splitter"></div>
          <div id="BLICK_iP_ImgColumn">
            <div id="BLICK_iP_loading">loading...</div>
          </div>`;
      this.pickerDiv.id = "BLICK_iconPicker";
      this.container.appendChild(this.pickerDiv);
      //
      this.imgCol = document.getElementById('BLICK_iP_ImgColumn');
      this.loading = document.getElementById('BLICK_iP_loading');
    }
    //
    insertStyle() {
      const sty = document.createElement("style");
      sty.id = 'BLICK_iconPicker_style';
      sty.innerHTML = `
          #BLICK_iconPicker {width: 400px;height: 200px;display: grid;grid-template-columns: 89px auto 1fr;overflow: hidden;background-color: #f8f8f8;user-select: none;flex:1;}
          #BLICK_iP_tabColumn {height:200px;display: grid;  grid-template-rows: auto 1fr;width:100%;padding: 1px;background-color: #e8e8e8;box-sizing: border-box;gap: 5px;}
          #BLICK_iP_splitter{background-color: #888;cursor: col-resize;width: 2px;}
          #BLICK_iP_seachInp {padding: 5px;font-size: 14px;border: 1px solid #ccc;border-radius: 5px;width:100%;box-sizing:border-box;}
          #BLICK_iP_tabs {display: flex;flex-direction: column;gap: 5px;overflow-y:auto}
          #BLICK_iP_tabs div {padding: 3px;background-color: #4CAF50;color: white;border: none;border-radius: 5px;cursor: pointer;}
          #BLICK_iP_tabs div.cur {background-color: #fdc03d;}
          #BLICK_iP_tabs div:hover {filter: brightness(90%);}
          #BLICK_iP_tabs div.dragover {background-color: #a5f0a9;}
          #BLICK_iP_tabs div[data-func] {text-align: center;}
          #BLICK_iP_ImgColumn {display: flex;grid-gap: 4px;padding: 4px;box-sizing: border-box;align-items: start;overflow-y: auto;width: 100%;flex-wrap: wrap;align-content: flex-start;}
          #BLICK_iP_ImgColumn .image-container {position: relative;display: flex;flex-direction: column;align-items: center;justify-content: center;}
          #BLICK_iP_ImgColumn .image-container img {width: 64px;height: 64px;border-radius: 5px;border: 2px solid #ddd;box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);object-fit: cover;}
          #imgContent .image-container:hover img {border-color: #4CAF50;}
          .selected {outline: 2px dashed #4CAF50;}`;
      document.head.appendChild(sty);
    }
    //
    async loadIcons() {
      const url = 'https://questden.org/kusaba/icons/thumb/';
      try {
        const response = await fetch(url);
        if (!response.ok) {
          throw new Error('Network response was not ok');
        }
        const html = await response.text();
        let match;
        const regex = /<td><a href="(.*?)"/g;
        while ((match = regex.exec(html)) !== null) {
          this.allIcons.push(match[1]);
        }
        this.allIcons.shift();
        this.searchItems();
      }
      catch (error) {
        console.error('There was a problem with the fetch operation:', error);
        this.searchItems();
      }
    }
    stripIconName(name) {
      if (name.length < 3)
        return "";
      return name.substring(0, name.lastIndexOf("s."));
    }
    setupEventListeners() {
      document.getElementById("BLICK_iP_ImgColumn")?.addEventListener('mousedown', this.onMouseDown.bind(this));
      document.getElementById("BLICK_iP_seachInp")?.addEventListener("input", this.searchItems.bind(this));
      document.getElementById("BLICK_iP_seachInp")?.addEventListener("keydown", (ev) => {
        if (ev.key === 'Enter')
          ev.preventDefault();
      });
      //
      document.getElementById('BLICK_iP_tabs')?.addEventListener("click", this.onTabClick.bind(this), true);
      document.getElementById("BLICK_iP_ImgColumn")?.addEventListener('click', (ev) => {
        let el = ev.target.closest(".image-container");
        if (el === null)
          return;
        this.clickCallback(":" + this.stripIconName(el.getAttribute("title") ?? "") + ":");
        //
      }, true);
      //
      const splitter = document.getElementById('BLICK_iP_splitter');
      splitter?.addEventListener('mousedown', (e) => {
        this.spliterDraggin = true;
        document.body.style.cursor = 'col-resize'; // Zeige den Resize-Cursor während des Ziehens
      });
      document.addEventListener('mousemove', (e) => {
        if (!this.spliterDraggin)
          return;
        const containerRect = this.pickerDiv.getBoundingClientRect();
        const newWidth = e.clientX - containerRect.left;
        if (newWidth > 50 && newWidth < containerRect.width - 50) {
          this.pickerDiv.style.gridTemplateColumns = `${newWidth}px auto 1fr`;
        }
      });
      //
      document.addEventListener('mouseup', () => {
        this.spliterDraggin = false;
        document.body.style.cursor = ''; // Cursor zurücksetzen
      });
      //
      const observer = new IntersectionObserver((entries) => {
        if (entries[0].isIntersecting && this.itemCount < this.shownIcons.length) {
          this.loadItems();
        }
      }, {
        root: this.imgCol,
        rootMargin: '0px',
        threshold: 1.0
      });
      if (this.loading !== null)
        observer.observe(this.loading);
    }
    //
    onMouseDown(event) {
      const target = event.target;
      if (target === null)
        return;
      const clst = target.closest('.image-container');
      //
      if (!this.selectedItems.includes(clst)) {
        this.selectedItems.forEach(item => item.classList.remove('selected'));
        this.selectedItems = [];
      }
      if (!clst) {
        this.startX = event.clientX + window.scrollX;
        this.startY = event.clientY + window.scrollY;
        //
        this.selectionBox = document.createElement('div');
        this.selectionBox.style.position = 'absolute';
        this.selectionBox.style.border = '2px dashed #4CAF50';
        this.selectionBox.style.backgroundColor = 'rgba(76, 175, 80, 0.2)';
        document.body.appendChild(this.selectionBox);
        //
        document.addEventListener('mousemove', this.onMouseMove.bind(this));
        document.addEventListener('mouseup', this.onMouseUp.bind(this));
      }
    }
    //
    onMouseMove(event) {
      const currentX = event.clientX + window.scrollX;
      const currentY = event.clientY + window.scrollY;
      if (!this.selectionBox)
        return;
      this.selectionBox.style.left = Math.min(this.startX, currentX) + 'px';
      this.selectionBox.style.top = Math.min(this.startY, currentY) + 'px';
      this.selectionBox.style.width = Math.abs(this.startX - currentX) + 'px';
      this.selectionBox.style.height = Math.abs(this.startY - currentY) + 'px';
      //
      const rect = this.selectionBox.getBoundingClientRect();
      if (rect === null)
        return;
      // Selektionslogik ändern, um nur hinzuzufügen, aber nicht sofort zu entfernen
      document.querySelectorAll('.image-container').forEach((item) => {
        const itemRect = item.getBoundingClientRect();
        //
        if (this.isIntersecting(rect, itemRect)) {
          // Bild nur hinzufügen, wenn es noch nicht selektiert ist
          if (!this.selectedItems.includes(item)) {
            item.classList.add('selected');
            this.selectedItems.push(item);
          }
        }
        else {
          if (this.selectedItems.includes(item)) {
            item.classList.remove('selected');
            this.selectedItems.splice(this.selectedItems.indexOf(item), 1);
          }
        }
      });
    }
    //
    onMouseUp() {
      document.removeEventListener('mousemove', this.onMouseMove.bind(this));
      document.removeEventListener('mouseup', this.onMouseUp.bind(this));
      //
      // Entferne das Auswahlrechteck nach der Selektion
      if (this.selectionBox && this.selectionBox.parentNode) {
        this.selectionBox.parentNode.removeChild(this.selectionBox);
        this.selectionBox = null;
      }
    }
    //
    isIntersecting(rect1, rect2) {
      return (rect1.left < rect2.right &&
        rect1.right > rect2.left &&
        rect1.top < rect2.bottom &&
        rect1.bottom > rect2.top);
    }
    //
    drag(event) {
      if (this.selectedItems.length === 0 && event.currentTarget !== null) {
        this.selectedItems.push(event.currentTarget);
      }
      event.dataTransfer?.setData('text/plain', JSON.stringify(this.selectedItems.map(item => item.title)));
    }
    //
    enableTabDrop() {
      const BLICK_iP_tabs = document.querySelectorAll('#BLICK_iP_tabs div');
      BLICK_iP_tabs.forEach(tab => {
        tab.addEventListener('dragover', (event) => {
          event.preventDefault();
          tab.classList.add('dragover');
        });
        //
        tab.addEventListener('dragleave', () => {
          tab.classList.remove('dragover');
        });
        //
        tab.addEventListener('drop', (event) => {
          event.preventDefault();
          tab.classList.remove('dragover');
          const titles = JSON.parse(event.dataTransfer?.getData('text') ?? "{}");
          const tabId = parseInt(tab.dataset.tabid ?? "-1");
          if (event.target?.dataset.func == "+")
            return;
          if (event.target?.dataset.role == "delete" || tabId === this.curActTab) { //remove entries
            if (confirm("Remove icons from list?")) {
              let ind = 0;
              titles.forEach((title) => {
                ind = this.editConf.data.iconList[tabId].arr.indexOf(title);
                if (ind > -1)
                  this.editConf.data.iconList[tabId].arr.splice(ind, 1);
              });
              this.clearItems();
            }
          }
          else { //add entries
            titles.forEach((title) => {
              if (!this.editConf.data.iconList[tabId].arr.includes(title)) {
                this.editConf.data.iconList[tabId].arr.push(title);
              }
            });
          }
          this.editConf.data.iconList[tabId].arr.sort();
          this.editConf.save();
        });
      });
    }
    //
    searchItems() {
      const search = document.getElementById("BLICK_iP_seachInp")?.value ?? "";
      //
      if (this.curActTab === -1 && search !== "") {
        this.shownIcons = this.allIcons.filter(el => (new RegExp(search.replace(/\s/g, ".*"), "i")).test(el));
      }
      else if (this.curActTab === -1 && search === "") {
        this.shownIcons = this.allIcons;
      }
      else if (this.curActTab !== -1 && search !== "") {
        this.shownIcons = this.editConf.data.iconList[this.curActTab].arr.filter(el => (new RegExp(search.replace(/\s/g, ".*"), "i")).test(el));
      }
      else if (this.curActTab !== -1 && search === "") {
        this.shownIcons = this.editConf.data.iconList[this.curActTab].arr;
      }
      //
      this.clearItems();
    }
    //
    clearItems() {
      if (this.imgCol === null)
        return;
      this.imgCol.innerHTML = "";
      this.imgCol.scrollTop = 0;
      this.itemCount = 0;
      this.loadItems();
    }
    //
    loadTabs() {
      const tabs = document.getElementById('BLICK_iP_tabs');
      if (tabs === null)
        return;
      this.editConf.data.iconList.forEach((item, index) => {
        const tab = document.createElement('div');
        tab.dataset.tabid = String(index);
        if (item.label == "Favourite" || item.label == "+") {
          tab.innerHTML = item.label;
          tab.dataset.func = item.label;
        }
        else {
          tab.innerHTML = item.label + " <span data-role='rename'>🖉</span><span data-role='delete'>🗑️</span>";
        }
        if (index === this.curActTab)
          tab.classList.add("cur");
        tabs.appendChild(tab);
      });
      this.enableTabDrop();
    }
    //
    updateTabs() {
      const tabs = document.getElementById('BLICK_iP_tabs');
      if (tabs === null)
        return;
      tabs.innerHTML = ''; // Leere die BLICK_iP_Tabs und erzeuge sie neu
      this.loadTabs();
    }
    //
    onTabClick(ev) {
      ev.preventDefault();
      ev.stopPropagation();
      let target = ev.target;
      if (target.dataset.func === "+") {
        const newLabel = prompt('New Name:', "");
        if (newLabel === "" || newLabel === null)
          return;
        this.editConf.data.iconList.splice(this.editConf.data.iconList.length - 1, 0, { label: newLabel, arr: [] });
        this.updateTabs();
      }
      else if (target.dataset.role === "rename") {
        const index = parseInt(target.parentElement?.dataset.tabid ?? "0");
        const newLabel = prompt('New Name:', this.editConf.data.iconList[index].label);
        if (newLabel === "" || newLabel === null)
          return;
        this.editConf.data.iconList[index].label = newLabel;
        this.updateTabs();
      }
      else if (target.dataset.role === "delete") {
        if (!confirm("Do you want to delete this icon list?"))
          return;
        const index = parseInt(target.parentElement?.dataset.tabid ?? "0");
        this.editConf.data.iconList.splice(index, 1);
        this.updateTabs();
      }
      else if (target.dataset.tabid !== undefined) {
        const index = parseInt(target.dataset.tabid);
        let stat = this.curActTab == index;
        document.querySelectorAll("#BLICK_iP_tabs>div.cur").forEach(el => el.classList.remove("cur"));
        if (!stat) {
          this.shownIcons = this.editConf.data.iconList[index].arr;
          target.classList.add("cur");
          this.curActTab = index;
        }
        else {
          this.shownIcons = this.allIcons;
          this.curActTab = -1;
        }
        this.searchItems();
      }
      this.editConf.save();
    }
    //
    loadItems() {
      if (this.imgCol === null || this.loading === null)
        return;
      const end = Math.min(this.itemCount + this.itemsPerLoad, this.shownIcons.length);
      for (let i = this.itemCount; i < end; i++) {
        this.imgCol.insertAdjacentHTML("beforeend", `
                <div class="image-container" draggable="true" title="${this.shownIcons[i]}">
                    <img src="https://questden.org/kusaba/icons/thumb/${this.shownIcons[i]}" alt="${this.shownIcons[i]}" />
                </div>`);
      }
      this.itemCount = end;
      this.imgCol.appendChild(this.loading);
      this.loading.style.display = (this.itemCount == this.shownIcons.length) ? "none" : "block";
      //
      document.querySelectorAll('.image-container').forEach(item => {
        item.addEventListener('dragstart', (ev) => {
          this.drag(ev);
        });
      });
    }
    //
    show() {
      this.pickerDiv.style.display = "";
    }
    hide() {
      this.pickerDiv.style.display = "none";
    }
    toggle() {
      if (this.isShown())
        this.show();
      else
        this.hide();
    }
    isShown() {
      return this.pickerDiv.style.display == "none";
    }
  }
  //
  class UndoRedoManager {
    //
    constructor(textarea) {
      this.undoStack = [];
      this.redoStack = [];
      this.textarea = textarea;
      this.applyingState = false;
      this.saveState();
      this.saveStateInterval = 500;
      this.saveStateTimer = 0;
      this.inputListener = this.debSaveState.bind(this);
      this.textarea.addEventListener('input', this.inputListener);
      this.keydownListener = (e) => {
        if (e.ctrlKey && e.key === 'z') {
          e.preventDefault();
          this.undo();
        }
        else if (e.ctrlKey && e.key === 'y') {
          e.preventDefault();
          this.redo();
        }
      };
      document.addEventListener('keydown', this.keydownListener);
    }
    debSaveState() {
      if (this.applyingState)
        return;
      clearTimeout(this.saveStateTimer);
      this.saveStateTimer = setTimeout(() => {
        this.saveState();
      }, this.saveStateInterval);
    }
    //
    saveState() {
      const state = {
        value: this.textarea.value,
        selectionStart: this.textarea.selectionStart,
        selectionEnd: this.textarea.selectionEnd
      };
      this.undoStack.push(state);
      this.redoStack = [];
    }
    //
    undo() {
      if (this.undoStack.length > 1) {
        const currentState = this.undoStack.pop();
        this.redoStack.push(currentState);
        const prevState = this.undoStack[this.undoStack.length - 1];
        this.applyState(prevState);
      }
    }
    //
    redo() {
      if (this.redoStack.length > 0) {
        const nextState = this.redoStack.pop();
        this.undoStack.push(nextState);
        this.applyState(nextState);
      }
    }
    //
    applyState(state) {
      this.applyingState = true;
      this.textarea.value = state.value;
      this.textarea.setSelectionRange(state.selectionStart, state.selectionEnd);
      const event = new Event('input', { bubbles: true });
      this.textarea.dispatchEvent(event);
      this.applyingState = false;
    }
    //
    destroy() {
      this.undoStack = [];
      this.redoStack = [];
      this.textarea.removeEventListener('input', this.inputListener);
      document.removeEventListener('keydown', this.keydownListener);
    }
  }
  //
  class ReplyForm {
    //
    constructor() {
      this.bar = null;
      this.textarea = null;
      this.colPickEl = null;
      this.iconPicker = null;
      this.buttonBar = null;
      this.extendBar = null;
      this.saveTimeStamp = null;
      this.genPrevTimer = 0;
      this.autoGeneratePrev = false;
      this.saveTimer = 0;
      this.undoManager = null;
      //
      this.editConf = new EditorConfig();
      this.editConf.load().then(this.insertStyle);
    }
    ;
    removeBar() {
      this.bar?.remove();
      document.querySelector("#BLICK_previewbut")?.remove();
      document.querySelector("#prevdiv")?.remove();
      this.undoManager?.destroy();
      this.bar = null;
    }
    ;
    insertStyle() {
      const sty = document.createElement("style");
      sty.id = "BLICK_stylebar";
      sty.innerHTML = `
        #BLICK_bigcont{background-color:#6f6f6f;width:100%;padding:3px 0;}
        .BLICK_style{display:inline-block;width:24px;height:24px;overflow:hidden;background-image: url('${imgRes.stylebuts}');background-repeat:no-repeat;}
        .BLICK_style:hover{background-position-y: -24px;}
        .BLICK_style:active{background-position-y: -48px;}
        #BLICK_pallette a {display: inline-block;	height: 20px;width: 20px; margin: 2px;background-color:attr(bgcol);}
        #BLICK_pallette {width: 145px;}
        #BLICK_colorpalette{padding:3px;box-sizing: border-box;display:none;}
        #BLICK_colorpalette.shown{display:flex;gap:10px;}
        #BLICK_cp{display:flex;height:100px;} 
        #BLICK_cp_colorDisplay { height: 40px; user-select: none;cursor:pointer;	}
        #BLICK_cp_bgDisplay { height: 40px; user-select: none;cursor:pointer;	} 
        #BLICK_cp_colTex {height: 20px;box-sizing: border-box;font-family: 'Courier New', Courier, monospace;text-transform: uppercase;border: 1px solid #ccc;border-radius: 4px;background-color: #f9f9f9;color: #333;text-align: center;z-index: 9;font-size: 11px;padding: 2px;width: 60px;letter-spacing: 0.5px;} 
        #BLICK_cp_colPrev { display: flex; flex-direction: column; width:60px;	} 
        #BLICK_cp .slider { writing-mode: vertical-lr; appearance: none; width: 10px; height: 100%; outline: none;	} 
        #BLICK_cp .slider::-webkit-slider-thumb { -webkit-appearance: none; width: 10px; height: 5px; background: #ffffff77; border: 1px solid black; border-radius: 2px; cursor: pointer; opacity: 0.8;	}
        #BLICK_cp .slider:hover::-webkit-slider-thumb { opacity: 1;} 
        #BLICK_cp .slider::-moz-range-thumb { width: 10px; height: 5px; background: #ffffff77; border: 1px solid black; border-radius: 2px; cursor: pointer; opacity: 0.8;	} 
        #BLICK_cp .slider:hover::-moz-range-thumb { opacity: 1;} 
        #BLICK_cp .slider[id="BLICK_cp_hue"] { background: linear-gradient(to bottom, hsl(0, 100%, 50%), hsl(60, 100%, 50%), hsl(120, 100%, 50%), hsl(180, 100%, 50%), hsl(240, 100%, 50%), hsl(300, 100%, 50%), hsl(360, 100%, 50%));	}
        #BLICK_colChoice a.customCol{border:1px dashed black;}
        #BLICK_colChoice a {display: inline-block;	height: 20px;width: 20px; margin: 2px;background-color:attr(bgcol);}
        #BLICK_colChoice {width: 130px;}
        #BLICK_extendBar{display:flex;}
        #BLICK_saveTimeStamp{position: absolute;bottom: 10px;right: 50px; font-size: 12px;color: #3333; pointer-events: none;}
        #BLICK_autoSaveBut{ padding: 0;float: right;margin: 0 10px;}
        #BLICK_autoSaveBut input, #BLICK_autoSaveBut label{ cursor: pointer;}
      `;
      document.head.appendChild(sty);
    }
    insertBar() {
      if (this.bar !== null)
        return;
      this.textarea = document.querySelector("form[name='postform'] textarea[name='message']");
      if (this.textarea === null) {
        this.bar = null;
        return;
      }
      //
      this.undoManager = new UndoRedoManager(this.textarea); //enough to work. ctrl+z manager
      //
      this.bar = document.createElement("div");
      this.bar.id = "BLICK_bigcont";
      this.textarea.parentElement.insertBefore(this.bar, this.textarea);
      this.textarea.parentElement.style.display = "block";
      //
      this.saveTimeStamp = document.createElement("div");
      this.saveTimeStamp.id = "BLICK_saveTimeStamp";
      this.textarea.parentElement.appendChild(this.saveTimeStamp);
      this.textarea.parentElement.style.position = "relative";
      //
      this.buttonBar = document.createElement("div");
      this.buttonBar.id = 'BLICK_buttonBar';
      this.bar.appendChild(this.buttonBar);
      this.extendBar = document.createElement("div");
      this.extendBar.id = 'BLICK_extendBar';
      this.bar.appendChild(this.extendBar);
      //
      let autosaveDiv = document.createElement("div");
      autosaveDiv.id = "BLICK_autoSaveBut";
      let autoSaveCB = document.createElement("input");
      autoSaveCB.type = "checkbox";
      autoSaveCB.id = "BLICK_autoSaveCB";
      autoSaveCB.checked = this.editConf.data.autoSave;
      autoSaveCB.addEventListener("change", (ev) => {
        let state = ev.target?.checked ?? true;
        this.editConf.set("autoSave", state);
        this.debsaveText();
      });
      autosaveDiv.appendChild(autoSaveCB);
      autosaveDiv.insertAdjacentHTML("beforeend", "<label for='BLICK_autoSaveCB'>autosave</label>");
      this.buttonBar.appendChild(autosaveDiv);
      //
      this.addButton("b", 0);
      this.addButton("u", 1);
      this.addButton("i", 2);
      this.addButton("s", 3);
      this.addButton("aa", 4);
      this.addButton("small", 5);
      this.addButton("code", 6);
      this.addButton("spoiler", 7);
      this.addButton("color", 10, () => {
        document.getElementById("BLICK_colorpalette")?.classList.toggle("shown");
        if (this.iconPicker)
          this.iconPicker.hide();
      });
      this.addButton("icon", 9, () => {
        if (this.extendBar === null)
          return;
        if (this.iconPicker === null) {
          this.iconPicker = new ImageManager(this.extendBar, this.editConf, (icon) => {
            if (this.textarea === null)
              return;
            let pos = this.textarea.selectionStart;
            this.textarea.value = this.textarea.value.substring(0, pos) + icon + this.textarea.value.substring(this.textarea.selectionEnd);
            this.textarea.setSelectionRange(pos + icon.length, pos + icon.length);
            this.textarea.focus();
            const event = new Event('input', { bubbles: true });
            this.textarea.dispatchEvent(event);
          });
        }
        else {
          this.iconPicker.toggle();
        }
        document.getElementById("BLICK_colorpalette")?.classList.remove("shown");
      });
      this.insertColorPallette();
      this.addPreviewBut();
      if (this.editConf.data.saveText !== "") {
        this.textarea.value = this.editConf.data.saveText;
      }
    }
    addPreviewBut() {
      const postForm = document.forms.namedItem("postform");
      if (!postForm)
        return;
      //
      let but = document.createElement("input");
      but.id = "BLICK_previewbut";
      but.type = "button";
      but.value = "Preview";
      //
      let subBut = document.querySelector("form[name='postform'] input[type='submit']");
      subBut?.parentElement?.insertBefore(but, subBut);
      subBut?.addEventListener("click", () => {
        this.editConf.set("saveText", "");
      });
      //
      let cbPrev = document.createElement("input");
      cbPrev.id = 'BLICK_cb_preview';
      cbPrev.type = "checkbox";
      cbPrev.style.marginRight = "-3px";
      cbPrev.title = "Automatically update the preview while typing.";
      but.parentElement?.insertBefore(cbPrev, but);
      //
      cbPrev.addEventListener("change", (ev) => {
        this.autoGeneratePrev = ev.target?.checked ?? false;
        this.generatePreview(ev);
      });
      but.addEventListener("click", this.generatePreview);
      document.querySelector("form[name='postform'] textarea[name='message']")?.addEventListener("input", () => {
        this.debGeneratePrev();
        this.debsaveText();
      });
      //
    }
    insertColorPallette() {
      let farbar = ["ff0000", "00ff00", "0000ff", "ffff00", "00ffff", "ff00ff"];
      let cp = document.createElement("div");
      cp.id = "BLICK_colorpalette";
      cp.className = "reply";
      //
      const texinh = { normal: "", brighter: "", darker: "", greys: "" };
      const greystep = 255.0 / (farbar.length - 1);
      let gry;
      for (let i = 0; i < farbar.length; i++) {
        texinh.normal += `<a href='#' bgcol='#${farbar[i]}' style='background-color:#${farbar[i]}');'></a>`;
        texinh.brighter += `<a href='#' bgcol='#${farbar[i].replace(/0/g, `8`)}' style='background-color:#${farbar[i].replace(/0/g, `8`)}');'></a>`;
        texinh.darker += `<a href='#' bgcol='#${farbar[i].replace(/f/g, `8`)}' style='background-color:#${farbar[i].replace(/f/g, `8`)}');'></a>`;
        gry = Math.floor(i * greystep).toString(16).padStart(2, '0');
        texinh.greys += `<a href='#' bgcol='#${gry + gry + gry}' style='background-color:#${gry + gry + gry}');'></a>`;
      }
      cp.innerHTML = `<div id='BLICK_pallette'>${Object.values(texinh).join("\n")}</div>`;
      //
      cp.addEventListener("click", (ev) => {
        if (!(ev.target instanceof HTMLAnchorElement))
          return;
        ev.preventDefault();
        ev.stopPropagation();
        this.styleSelection("color=" + ev.target.getAttribute("bgcol"));
      }, true);
      //
      this.extendBar?.appendChild(cp);
      this.insertColorPicker();
      this.insertColorChoice();
    }
    insertColorPicker() {
      let pick = document.createElement("div");
      pick.id = "BLICK_cp";
      pick.innerHTML = `
          <div id="BLICK_cp_colPrev">
            <div id="BLICK_cp_colorDisplay">Click to insert</div>
            <div id="BLICK_cp_bgDisplay"></div>
            <input type="text" id="BLICK_cp_colTex" />
          </div>
          <input type="range" id="BLICK_cp_hue" class="slider" min="0" max="360" value="0" step="1" orient="vertical">
          <input type="range" id="BLICK_cp_saturation" class="slider" min="0" max="100" value="0" step="10"
            orient="vertical">
          <input type="range" id="BLICK_cp_lightness" class="slider" min="0" max="100" value="50" step="5" orient="vertical">`;
      //
      document.getElementById("BLICK_colorpalette")?.appendChild(pick);
      //
      this.colPickEl = {
        hue: document.getElementById('BLICK_cp_hue'),
        sat: document.getElementById('BLICK_cp_saturation'),
        lum: document.getElementById('BLICK_cp_lightness'),
        bgCol: document.getElementById('BLICK_cp_bgDisplay'),
        prevCol: document.getElementById('BLICK_cp_colorDisplay'),
        prevText: document.getElementById('BLICK_cp_colTex')
      };
      //
      this.colPickEl.hue.addEventListener("input", () => { this.updatePickerColer(true); });
      this.colPickEl.sat.addEventListener("input", () => { this.updatePickerColer(true); });
      this.colPickEl.lum.addEventListener("input", () => { this.updatePickerColer(true); });
      this.colPickEl.prevText.addEventListener('input', () => {
        if (this.colPickEl == null)
          return;
        this.colPickEl.prevCol.style.color = this.colPickEl.prevText.value;
        this.colPickEl.bgCol.style.backgroundColor = this.colPickEl.prevCol.style.color;
        let rgb = this.colPickEl.prevCol.style.color.match(/(\d+)/g) ?? ["0", "0", "0"];
        let hsl = rgb2hsl(parseInt(rgb[0]), parseInt(rgb[1]), parseInt(rgb[2]));
        this.colPickEl.hue.value = String(hsl[0]);
        this.colPickEl.sat.value = String(100 - hsl[1]);
        this.colPickEl.lum.value = String(100 - hsl[2]);
        this.updatePickerColer(false);
      });
      let insertColFun = () => {
        if (this.colPickEl === null)
          return;
        this.styleSelection("color=" + this.colPickEl.prevText.value);
      };
      this.colPickEl.prevCol.addEventListener("click", insertColFun);
      this.colPickEl.bgCol.addEventListener("click", insertColFun);
      //
      this.updatePickerColer(true);
    }
    updatePickerColer(textupdate = false) {
      if (this.colPickEl == null)
        return;
      const hue = this.colPickEl.hue.value;
      const saturation = 100 - parseInt(this.colPickEl.sat.value);
      const lightness = 100 - parseInt(this.colPickEl.lum.value);
      //
      this.colPickEl.prevCol.style.color = `hsl(${hue}, ${saturation}%, ${lightness}%)`;
      this.colPickEl.bgCol.style.backgroundColor = this.colPickEl.prevCol.style.color;
      if (textupdate) {
        this.colPickEl.prevText.value = "#" + (this.colPickEl.prevCol.style.color.match(/(\d+)/g) ?? ["0", "0", "0"]).map(el => parseInt(el).toString(16).padStart(2, "0")).join("");
      }
      this.colPickEl.sat.style.background = `linear-gradient(to top,hsl(${hue}, 0%, 50%),hsl(${hue}, 100%, 50%))`;
      this.colPickEl.lum.style.background = `linear-gradient(to top,hsl(${hue}, ${saturation}%, 0%),hsl(${hue}, ${saturation}%, 50%),hsl(${hue}, ${saturation}%, 100%))`;
    }
    insertColorChoice() {
      let farbar = this.editConf.data.colorList;
      let cp = document.getElementById("BLICK_colorpalette");
      if (cp === null)
        return;
      // 
      let colCh = document.createElement("div");
      colCh.id = "BLICK_colChoice";
      let texinh = "<div>Rightclick to set</div>";
      for (let i = 0; i < farbar.length; i++) {
        texinh += `<a href='#' class='customCol' customId='${i}' bgcol='${farbar[i]}' style='background-color:${farbar[i]}');'></a>`;
      }
      colCh.innerHTML = texinh;
      cp.appendChild(colCh);
      //
      cp.addEventListener("contextmenu", (ev) => {
        if (!(ev.target instanceof HTMLAnchorElement) || ev.target.className !== "customCol" || this.colPickEl === null)
          return;
        ev.preventDefault();
        ev.stopPropagation();
        let id = parseInt(ev.target.getAttribute("customId") ?? "0");
        this.editConf.data.colorList[id] = this.colPickEl.prevText.value;
        ev.target.setAttribute("bgcol", this.colPickEl.prevText.value);
        ev.target.style.backgroundColor = this.colPickEl.prevText.value;
        this.editConf.save();
      }, true);
    }
    addButton(tag, bgpos, callback = null) {
      const butB = document.createElement("div");
      butB.className = "BLICK_style";
      butB.title = `[${tag}]`;
      butB.addEventListener("click", () => {
        if (callback === null)
          this.styleSelection(tag);
        else
          callback();
      });
      butB.style.backgroundPositionX = -24 * bgpos + "px";
      this.buttonBar?.appendChild(butB);
    }
    //
    styleSelection(tag) {
      if (this.textarea === null)
        return;
      let texStart = this.textarea.selectionStart ?? 0;
      let texEnd = this.textarea.selectionEnd ?? 0;
      let pureTag = tag.indexOf("=") === -1 ? tag : tag.substring(0, tag.indexOf("=")); //strip parameters from closing tag
      //
      if (texStart >= tag.length + 2 && this.textarea.value.substring(texStart - tag.length - 2, texStart) == "[" + tag + "]") {
        //remove tag if selection preceeded by tag
        this.textarea.value = this.textarea.value.substring(0, texStart - tag.length - 2) +
          this.textarea.value.substring(texStart, texEnd) +
          this.textarea.value.substring(texEnd + pureTag.length + 3);
        this.textarea.selectionStart = texStart - tag.length - 2; //selection shifted by start tag length
        this.textarea.selectionEnd = texEnd - tag.length - 2;
      }
      else if (pureTag == "color" && this.textarea.value.substring(texStart - "color=#ffffff]".length, texStart - "=#ffffff]".length) == "color") {
        //color change on color tag if new color is different
        this.textarea.value = this.textarea.value.substring(0, texStart - tag.length - 2) + "[" + tag + "]" +
          this.textarea.value.substring(texStart, texEnd) + "[/" + pureTag + "]" +
          this.textarea.value.substring(texEnd + pureTag.length + 3);
        this.textarea.selectionStart = texStart; //original selection
        this.textarea.selectionEnd = texEnd;
      }
      else {
        //add tag
        this.textarea.value = this.textarea.value.substring(0, texStart) +
          "[" + tag + "]" + this.textarea.value.substring(texStart, texEnd) + "[/" + pureTag + "]" +
          this.textarea.value.substring(texEnd);
        this.textarea.selectionStart = texStart + tag.length + 2; //selection shifted by start tag length
        this.textarea.selectionEnd = texEnd + tag.length + 2;
      }
      this.textarea.focus();
      //
      const event = new Event('input', { bubbles: true });
      this.textarea.dispatchEvent(event);
    }
    debsaveText() {
      if (!this.editConf.data.autoSave) {
        if (this.saveTimeStamp !== null)
          this.saveTimeStamp.innerText = "";
        this.editConf.set("saveText", "");
        return;
      }
      clearTimeout(this.saveTimer);
      this.saveTimer = setTimeout(() => {
        this.saveText(null);
      }, 500);
    }
    saveText(ev) {
      ev?.preventDefault();
      this.editConf.set("saveText", this.textarea?.value ?? "");
      //
      const now = new Date();
      const year = now.getFullYear();
      const month = String(now.getMonth() + 1).padStart(2, '0'); // Monate von 0 bis 11
      const day = String(now.getDate()).padStart(2, '0');
      const hours = String(now.getHours()).padStart(2, '0');
      const minutes = String(now.getMinutes()).padStart(2, '0');
      const seconds = String(now.getSeconds()).padStart(2, '0');
      //
      const formattedDate = `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;
      if (this.saveTimeStamp !== null)
        this.saveTimeStamp.innerText = `Last saved: ${formattedDate}`;
    }
    debGeneratePrev() {
      if (!this.autoGeneratePrev)
        return;
      clearTimeout(this.genPrevTimer);
      this.genPrevTimer = setTimeout(() => {
        this.generatePreview(null);
      }, 500);
    }
    generatePreview(ev) {
      ev?.preventDefault();
      //
      const postForm = document.forms.namedItem("postform");
      if (postForm === null)
        return;
      let prevDiv = document.getElementById('prevdiv');
      //
      if (document.getElementById('prevdiv') === null) {
        prevDiv = document.createElement('div');
        prevDiv.id = 'prevdiv';
        prevDiv.className = 'reply';
        prevDiv.style.textAlign = 'left';
        prevDiv.style.backgroundColor = '';
        postForm.appendChild(prevDiv);
      }
      //
      const boardValue = postForm.elements.namedItem("board")?.value ?? "";
      const replyThreadValue = postForm.elements.namedItem("replythread")?.value ?? "";
      const messageValue = postForm.elements.namedItem("message")?.value ?? "";
      //
      postpreview('prevdiv', boardValue, replyThreadValue, messageValue); //defined in website.
    }
  }
  ;
  ;
  class WatchData extends Storage {
    constructor() {
      let def = {
        numLinkMode: 1,
        threads: new Map(),
        default: {
          label: "Default",
          author: "none",
          section: "general",
          highImgOnly: true,
          highIDs: [],
          highNames: [],
          ignoreIDs: [],
          ignoreNames: [],
          lastReadId: "",
          currentReadId: "",
          newEntrCnt: 0,
          totalEntrCnt: 0,
        }
      };
      super("watchbar", def);
    }
  }
  ;
  //
  class WatchBar {
    constructor(sets) {
      this.curThread = null;
      this.saveTimer = 0;
      this.saveInterval = 500;
      this.thList = ["draw", "meep", "quest", "questdis", "tg", "questarch"];
      this.sets = sets;
      this.wdata = new WatchData();
      this.wdata.load().then(() => {
        this.insertHTML();
        this.insertStyle();
        this.insertObserver();
        this.insertEntries();
        this.insertSettingHTML();
      });
    }
    destroy() {
      document.getElementById("BLICK_watch_bar")?.remove();
      document.getElementById("BLICK_stylewatchbar")?.remove();
      document.getElementById("BLICK_wbar_form-container")?.remove();
    }
    insertHTML() {
      const wbar = document.createElement("div"); //hover notch&container with overflow-scroll
      wbar.id = "BLICK_watch_bar";
      //
      this.cont = document.createElement("div"); // inner container containing items
      this.cont.id = "BLICK_watch_wrap";
      wbar.appendChild(this.cont);
      //
      const updbut = document.createElement("img");
      updbut.id = "BLICK_watch_update";
      updbut.title = "Check for updates";
      updbut.className = "BLICK_wbar_headButs";
      updbut.src = imgRes.refreshimg;
      wbar.appendChild(updbut);
      updbut.addEventListener("click", () => { this.wdata.load().then(() => { this.updateEntries(); }); });
      //
      const defaultSetBut = document.createElement("button");
      defaultSetBut.id = "BLICK_watch_defaultSet";
      defaultSetBut.title = "Set default settings.";
      defaultSetBut.className = "BLICK_wbar_headButs";
      defaultSetBut.innerText = "🛠️";
      wbar.appendChild(defaultSetBut);
      defaultSetBut.addEventListener("click", () => { this.showSettingDiag("", true); });
      //
      const importBut = document.createElement("button");
      importBut.id = "BLICK_watch_import";
      importBut.title = "Import from questden.org";
      importBut.className = "BLICK_wbar_headButs";
      importBut.innerText = "📥";
      wbar.appendChild(importBut);
      importBut.addEventListener("click", () => { this.fetchFromSite(); });
      //
      const exportBut = document.createElement("button");
      exportBut.id = "BLICK_watch_export";
      exportBut.title = "Export to questden.org";
      exportBut.className = "BLICK_wbar_headButs";
      exportBut.innerText = "📤";
      wbar.appendChild(exportBut);
      exportBut.addEventListener("click", () => { this.copyToSite(); });
      //
      wbar.addEventListener("click", (ev) => {
        if (!(ev.target instanceof HTMLElement))
          return;
        if (ev.target.dataset.role == "remove") {
          let row = ev.target.closest("div.BLICK_wrow");
          if (row === null || row.dataset.id === undefined)
            return;
          if (!confirm(`Remove entry for ${this.wdata.data.threads.get(row.dataset.id)?.label}?`))
            return;
          this.wdata.data.threads.delete(row.dataset.id ?? "0");
          this.wdata.save();
          row.remove();
          ev.preventDefault();
          ev.stopPropagation();
        }
        else if (ev.target.dataset.role == "options") {
          let row = ev.target.closest("div.BLICK_wrow");
          if (row === null || row.dataset.id === undefined)
            return;
          this.showSettingDiag(row.dataset.id);
          ev.preventDefault();
          ev.stopPropagation();
        }
      }, true);
      document.body.appendChild(wbar);
    }
    copyToSite() {
      const fetchPromises = [];
      this.wdata.data.threads.forEach((thread, id) => {
        fetchPromises.push(fetch(`https://questden.org/kusaba/threadwatch.php?do=addthread&board=${thread.section}&threadid=${id}`).then((ret) => {
          if (ret.ok)
            console.log(`added ${thread?.label}, ${thread?.section}/${id} to your watchlist.`);
        }));
      });
      Promise.all(fetchPromises).then(() => {
        getwatchedthreads('0', "quest");
        alert(`${this.wdata.data.threads.size} threads exported.`);
      });
    }
    insertSettingHTML() {
      this.setDiag = document.createElement("div");
      this.setDiag.id = "BLICK_wbar_set_overlay";
      this.setDiag.style.display = "none";
      this.setDiag.innerHTML = `
      <div id="BLICK_wbar_form-container">
        <h2>Settings</h2>
        <p id="BLICK_wbar_formSubtitle">Options for 'Divequest ch4'</p>
        <div id="BLICK_wbar_form-grid">
        <div class="BLICK_wbar_setSect" id='BLICK_wbar_genSec'>
          <span class='BLICK_wbar_secTitl'>General Settings</span>
            <label for="BLICK_wbar_autoUpdate">Automatically update:</label>
            <div class="checkbox-container">
              <input type="checkbox" id="BLICK_wbar_autoUpdate">
              <label for="BLICK_wbar_autoUpdate">On pageload</label>
            </div>
            <label for="BLICK_wbar_linkTo">Number links to:</label>
            <select id="BLICK_wbar_linkTo">
              <option value="new">Last known post</option>
              <option value="current">Last reading position</option>
              <option value="none">No specific post</option>
            </select>
          </div>
          <div class="BLICK_wbar_setSect" id='BLICK_wbar_threadSec'>
            <span class='BLICK_wbar_secTitl'>Default Thread Settings</span>
            <label for="BLICK_wbar_highlightNew">Count New:</label>
            <div class="checkbox-container">
              <input type="checkbox" id="BLICK_wbar_highlightNew">
              <label for="BLICK_wbar_highlightNew">Only Images</label>
            </div>
            <label for="BLICK_wbar_highlightNames">Only Names</label>
            <input type="text" id="BLICK_wbar_highlightNames" title="Names, comma separated">
            <label for="BLICK_wbar_highlightIDs">Only IDs</label>
            <input type="text" id="BLICK_wbar_highlightIDs" title="IDs, comma separated">
            <label for="BLICK_wbar_ignoreNames">Ignore Names</label>
            <input type="text" id="BLICK_wbar_ignoreNames" title="Names, comma separated">
            <label for="BLICK_wbar_ignoreIDs">Ignore IDs</label>
            <input type="text" id="BLICK_wbar_ignoreIDs" title="IDs, comma separated">
          </div>
        </div>
        <div id="BLICK_wbar_button-row">
          <button data-role="save">Save</button>
          <button data-role="cancel">Cancel</button>
          <button data-role="default">Default</button>
          <button data-role="overwrite">Apply to all</button>
        </div>
      </div>`;
      document.body.appendChild(this.setDiag);
      this.setDiag.addEventListener("click", function (ev) {
        if (ev.target == this)
          this.style.display = "none"; //this in function() refers to "this.setDiag"
      });
      document.getElementById("BLICK_wbar_button-row")?.addEventListener("click", (ev) => {
        let target = ev.target;
        if (target.dataset.role == null || this.setDiag.dataset.threadid === undefined)
          return;
        let id = this.setDiag.dataset.threadid;
        switch (target.dataset.role) {
          case "save":
            this.saveSettingDiag(id);
          default:
          case "cancel":
            this.setDiag.style.display = "none";
            return;
          case "default":
            this.showSettingDiag(id, true);
            break;
          case "overwrite":
            if (confirm("Set settings of other entries to this?")) {
              this.wdata.data.threads.forEach((thread, tid) => { this.saveSettingDiag(tid); });
              if (id === "")
                this.saveSettingDiag("");
              this.setDiag.style.display = "none";
            }
            break;
        }
        ev.preventDefault();
        ev.stopPropagation();
      }, true);
    }
    insertStyle() {
      const sty = document.createElement("style");
      sty.id = "BLICK_stylewatchbar";
      sty.innerHTML = `
        #BLICK_watch_bar{width:20px;overflow-y:auto;overflow-x:hidden;position:fixed;transition:width 0.5s,opacity 0.5s, max-height 0.5s, border-radius 1s; right:0px;max-height:30px;height:30px;background-color:#fad;z-index:8;opacity:0.5;top:85px;border:2px ridge #ddf;border-top-left-radius:25px;border-bottom-left-radius:25px;font-size: 16px;color: #d00;}
        #BLICK_watch_bar:hover{overflow-y:auto;overflow-x:hidden;opacity:1;width:400px;height:auto;border-bottom-left-radius:5px;border-top-left-radius:5px;max-height:calc(95vh - 85px)}
        #BLICK_watch_bar>*{visibility:hidden;cursor:default;}
        #BLICK_watch_bar:hover>*{visibility:visible;}
        #BLICK_watch_bar .BLICK_wbar_headButs{width:16px;height:16px;z-index:99;position:absolute;top:0;cursor:pointer;border:none;background:none;}
        #BLICK_watch_bar #BLICK_watch_update{right:0;}
        #BLICK_watch_bar #BLICK_watch_defaultSet{right:25px;}
        #BLICK_watch_bar #BLICK_watch_import{right:50px;}
        #BLICK_watch_bar #BLICK_watch_export{right:75px;}
        #BLICK_watch_bar .BLICK_wbar_headButs:hover{filter:brightness(120%);}
        #BLICK_watch_bar button:hover{filter:brightness(120%);}
        #BLICK_watch_wrap{display: grid; grid-template-columns: auto 100px 45px 50px;}
        #BLICK_watch_bar .BLICK_wrow {display: contents; padding: 0px 5px;}
        #BLICK_watch_bar .BLICK_wrow:nth-child(2n)>div.BLICK_wcell{background-color: #F0D0C6;}
        #BLICK_watch_bar .BLICK_wrow:nth-child(2n+1)>div.BLICK_wcell{background-color: #FBC1AF;}
        #BLICK_watch_bar .BLICK_wrow>div.BLICK_wcell{display: inline-block; vertical-align: top; word-wrap: break-word;line-height:23px;padding:0 5px;}
        #BLICK_watch_bar .BLICK_whead{background-color: #EEAA88;grid-column: 1 / -1;text-align: center;font-weight: bold;padding: 2px;color: black;}
        #BLICK_watch_bar .BLICK_wactions button{width:16px;height:16px;font-size:14px;border:none;background:none;cursor:pointer;display:inline-block;}
        #BLICK_watch_bar a:hover{cursor:pointer;}
        #BLICK_watch_bar div.BLICK_wauthor{color: green;font-weight: bold;}
        #BLICK_watch_bar div.BLICK_wnew a{color: blue;}
        #BLICK_watch_bar div.BLICK_title a{color: red;}
        #BLICK_wbar_set_overlay {position: fixed;top: 0;left: 0;width: 100vw;height: 100vh;background-color: rgba(0, 0, 0, 0.5);display: flex;justify-content: center;align-items: center;z-index: 1000;}
        #BLICK_wbar_form-container{background-color: #fffff0;padding: 20px;border-radius: 8px;box-shadow: 2px 2px 5px rgba(0, 0, 0, 0.2);max-width: 400px;width: 90%;font-family: Arial, sans-serif;border: 1px solid #b0a394;}
        #BLICK_wbar_form-container h2 {margin-top: 0;font-size: 1.4em;color: #333;text-align: center;}
        #BLICK_wbar_form-container p {font-size: 0.9em;color: #666;text-align: center;margin-top: 0;margin-bottom: 15px;}
        #BLICK_wbar_form-grid {margin-bottom: 15px;}
        #BLICK_wbar_form-grid label {font-size: 0.9em;color: #555;display: flex;align-items: center;justify-content: flex-end;}
        #BLICK_wbar_form-grid input[type="text"] {padding: 8px;border: 1px solid #ccc;border-radius: 4px;font-size: 0.9em;}
        #BLICK_wbar_form-grid input[type="checkbox"] {cursor:pointer}
        #BLICK_wbar_form-grid .checkbox-container {display: flex;align-items: center;justify-content: center;}
        #BLICK_wbar_form-grid .checkbox-container label {margin-left: 8px;cursor:pointer}
        #BLICK_wbar_button-row {display: flex;gap: 10px;justify-content: space-between;}
        #BLICK_wbar_button-row button {flex: 1;padding: 8px;border: none;border-radius: 4px;font-size: 0.9em;cursor: pointer;transition: background-color 0.2s;color: white;}
        #BLICK_wbar_button-row button:hover {opacity: 0.9;}
        #BLICK_wbar_button-row button[data-role="save"] { background-color: #3ba44b;}
        #BLICK_wbar_button-row button[data-role="cancel"] { background-color: #bb5546;}
        #BLICK_wbar_button-row button[data-role="default"] { background-color: #304f79;}
        #BLICK_wbar_button-row button[data-role="overwrite"] { background-color: #dd9243;}
        .BLICK_wbutton_iswatched{transition:filter 0.5s;filter: hue-rotate(150deg) brightness(150%);}
        .BLICK_wbar_setSect{position: relative;margin-bottom: 15px; display: grid;grid-template-columns: 150px auto;grid-column: span 2;gap: 15px;border: 1px solid #aaa;border-radius: 5px;padding: 15px;padding-top:25px;}
        #BLICK_wbar_linkTo {padding: 10px; border: 1px solid #ccc; border-radius: 4px; box-sizing: border-box;}
        .BLICK_wbar_secTitl{position: absolute;left: 5px;right: 0;font-size: smaller;color: #aaa;}
      `;
      document.head.appendChild(sty);
    }
    showSettingDiag(id, def = false) {
      if (this.setDiag === null || document.getElementById("BLICK_wbar_formSubtitle") == null ||
        (id !== "" && !this.wdata.data.threads.has(id)))
        return;
      const thread = def ? this.wdata.data.default : this.wdata.data.threads.get(id);
      //
      if (id === "") {
        document.getElementById("BLICK_wbar_formSubtitle").style.display = "none";
        document.getElementById("BLICK_wbar_genSec").style.display = "";
        document.querySelector("#BLICK_wbar_threadSec .BLICK_wbar_secTitl").innerText = "Default Thread settings";
        document.getElementById("BLICK_wbar_autoUpdate").checked = this.sets.data.autoUpdWbar;
        document.getElementById("BLICK_wbar_linkTo").selectedIndex = this.wdata.data.numLinkMode;
      }
      else {
        document.getElementById("BLICK_wbar_formSubtitle").style.display = "";
        document.getElementById("BLICK_wbar_formSubtitle").innerText = this.wdata.data.threads.get(id)?.label ?? "";
        document.getElementById("BLICK_wbar_genSec").style.display = "none";
        document.querySelector("#BLICK_wbar_threadSec .BLICK_wbar_secTitl").innerText = "Thread settings";
      }
      //
      document.getElementById("BLICK_wbar_highlightNew").checked = thread?.highImgOnly ?? true;
      document.getElementById("BLICK_wbar_highlightNames").value = thread?.highNames.join(",") ?? "";
      document.getElementById("BLICK_wbar_highlightIDs").value = thread?.highIDs.join(",") ?? "";
      document.getElementById("BLICK_wbar_ignoreNames").innerText = thread?.ignoreNames.join(",") ?? "";
      document.getElementById("BLICK_wbar_ignoreIDs").innerText = thread?.ignoreIDs.join(",") ?? "";
      this.setDiag.style.display = "flex";
      this.setDiag.dataset.threadid = id;
    }
    saveSettingDiag(id, close = false) {
      if (this.setDiag === null || document.getElementById("BLICK_wbar_formSubtitle") == null ||
        (id !== "" && !this.wdata.data.threads.has(id)))
        return;
      const thread = (id !== "") ? this.wdata.data.threads.get(id) : this.wdata.data.default;
      //
      let sanit = (str) => { return str.split(",").map(name => name.trim()).filter(name => name !== ""); };
      thread.highImgOnly = document.getElementById("BLICK_wbar_highlightNew").checked ?? true;
      thread.highNames = sanit(document.getElementById("BLICK_wbar_highlightNames").value);
      thread.highIDs = sanit(document.getElementById("BLICK_wbar_highlightIDs").value);
      thread.ignoreNames = sanit(document.getElementById("BLICK_wbar_ignoreNames").value);
      thread.ignoreIDs = sanit(document.getElementById("BLICK_wbar_ignoreIDs").value);
      if (id === "") {
        this.sets.set("autoUpdWbar", document.getElementById("BLICK_wbar_autoUpdate").checked ?? true);
        this.wdata.set("numLinkMode", document.getElementById("BLICK_wbar_linkTo").selectedIndex ?? 0);
      }
      //
      this.debSave();
      if (close)
        this.setDiag.style.display = "none";
    }
    debSave() {
      clearTimeout(this.saveTimer);
      this.saveTimer = setTimeout(() => {
        this.wdata.save();
        this.sets.save();
      }, this.saveInterval);
    }
    scrollObserver(entries, observer) {
      let topElement = document.elementFromPoint(300, 0); //get top element at 300px to left
      if (!topElement || topElement.tagName === "HTML")
        topElement = document.elementFromPoint(300, 20); //if gap, check 20px below
      topElement = topElement?.closest('td.reply') ?? null;
      if (topElement !== null) {
        let postId = topElement.id.match(/\d+/); //id of element at top of viewport
        if (this.curThread != null && postId != null) { //if watched
          this.curThread.currentReadId = postId[0]; //
          this.debSave();
        }
      }
    }
    addThread(id, board) {
      if (this.wdata.data.threads.has(id))
        return;
      const nthread = { ...this.wdata.data.default };
      nthread.section = board;
      this.wdata.data.threads.set(id, nthread);
      this.insertEntries();
      this.updateEntries();
    }
    insertObserver() {
      if (this.sets.data.autoUpdWbar)
        this.updateEntries(); //autoupdate on pageload
      document.querySelectorAll("img.watchthread").forEach((el) => {
        const clickstr = el.parentElement?.getAttribute("onclick");
        const threadData = clickstr?.match(/addtowatchedthreads\('(.+?)','(.+?)'\)/) ?? null;
        if (threadData === null || threadData.length < 3)
          return;
        el.dataset.targetid = threadData[1];
        el.dataset.targetboard = threadData[2];
        if (this.wdata.data.threads.has(threadData[1]))
          el.classList.add("BLICK_wbutton_iswatched");
        //
        el.addEventListener("click", (ev) => {
          let target = ev.target;
          if (target.dataset.targetid === undefined || target.dataset.targetboard === undefined)
            return;
          this.addThread(target.dataset.targetid, target.dataset.targetboard);
          target.classList.add("BLICK_wbutton_iswatched");
        });
      });
      //
      const locMatch = location.href.match(/https:\/\/questden.org\/kusaba\/(\w+)\/res\/(\d+)(\+\d+)?.html/i);
      if (locMatch === null)
        return;
      let id = locMatch[2];
      if (!this.wdata.data.threads.has(id))
        return; //log data only for watched websites.
      this.curThread = this.wdata.data.threads.get(id) ?? null;
      //scroll to last read post
      let currentReadPost = document.getElementById(`reply${this.curThread?.currentReadId}`);
      if (currentReadPost)
        currentReadPost.scrollIntoView(); //executed before Quest Reader, so should not collide.
      //
      const replyElements = document.querySelectorAll('td.reply');
      const lastmatch = replyElements[replyElements.length - 1].id.match(/\d+/);
      if (lastmatch !== null && this.curThread !== null)
        this.curThread.lastReadId = lastmatch[0];
      this.wdata.save();
      //
      const observer = new IntersectionObserver((a, b) => { this.scrollObserver(a, b); }, {
        root: null, //  Viewport
        rootMargin: '0px',
        threshold: 0 // Pixel
      });
      replyElements.forEach(el => observer.observe(el));
    }
    fetchFromSite() {
      const fetchPromises = this.thList.map(el => {
        return fetch(`https://questden.org/kusaba/threadwatch.php?board=${el}`)
          .then(response => response.text())
          .then(htmlText => {
            const parser = new DOMParser();
            const doc = parser.parseFromString(htmlText, 'text/html');
            const entries = doc.querySelectorAll('body span.filetitle');
            entries.forEach(entry => {
              const id = entry.previousElementSibling?.textContent?.trim() ?? null;
              if (id === null)
                return;
              const entr = { ...this.wdata.data.default };
              entr.section = el;
              entr.label = entry.textContent?.trim() ?? "";
              entr.author = entry.nextElementSibling?.textContent?.trim() ?? "";
              const replyEl = entry.nextElementSibling?.nextElementSibling;
              entr.newEntrCnt = parseInt(replyEl?.textContent?.trim() ?? "0");
              //link to newest element.
              //however, lastread should be the element before that
              //also, currentread should be the one to jump to when visiting.
              // if(replyEl?.tagName==="a"){
              // 	const mat=replyEl.getAttribute("href")?.match(/\d+$/)??null;
              // if(mat!==null)entr.lastReadId=parseInt(mat[0]);
              // }
              if (!this.wdata.data.threads.has(id)) {
                this.wdata.data.threads.set(id, entr);
              }
            });
          });
      });
      Promise.all(fetchPromises)
        .then(() => {
          // this.wdata.data.threads.sort((a, b) => a.section.localeCompare(b.section));
          this.wdata.save();
          this.insertEntries();
        })
        .catch(error => {
          console.error("Fehler beim Laden der Daten:", error);
        });
    }
    generateLink(thread, id, curReadInd) {
      let add = "";
      let jump = "";
      let targetInd = this.wdata.data.numLinkMode == 1 ? thread.newEntrCnt : (thread.totalEntrCnt - curReadInd);
      //target last known or last read depending on mode
      if (targetInd < 50 && thread.totalEntrCnt > 50)
        add = "+50";
      else if (targetInd < 100 && thread.totalEntrCnt > 100)
        add = "+100";
      //+50/+100 if possible and target in page.
      if (curReadInd == -1 && this.wdata.data.numLinkMode == 1)
        add = "";
      //when inserting without update, knowing last reading position is within +50/+500 is not possible.
      if (this.wdata.data.numLinkMode == 0 && thread.lastReadId !== "")
        jump = "#" + thread.lastReadId;
      if (this.wdata.data.numLinkMode == 1 && thread.currentReadId !== "")
        jump = "#" + thread.currentReadId;
      return `https://questden.org/kusaba/${thread.section}/res/${id}${add}.html${jump}`;
    }
    updateEntries() {
      let fetchPromise = [];
      document.getElementById("BLICK_watch_update")?.style.setProperty("filter", "hue-rotate(150deg)");
      this.wdata.data.threads.forEach((thread, id) => {
        fetchPromise.push(fetch(`https://questden.org/kusaba/quest/res/${id}.html`)
          .then(response => response.text())
          .then(htmlText => {
            //performance improve idea: count instances of reply update in text for total count. Then cut away text in front of lastReadId before parsing.
            const parser = new DOMParser();
            const doc = parser.parseFromString(htmlText, 'text/html');
            //
            //parse website
            let lastReadElement = null;
            let newCount = 0;
            let curReadInd = -1; //index of current read position
            const replyElements = doc.querySelectorAll(`td.reply`);
            replyElements.forEach((td, ind) => {
              if (thread.currentReadId == td.id)
                curReadInd = ind;
              if (lastReadElement === null && (td.id.endsWith(thread.lastReadId) || thread.lastReadId == ""))
                lastReadElement = td; //filter for only new elements
              if (lastReadElement !== null) { //ignore old
                let postauthName = td.querySelector('span.postername')?.textContent?.toLowerCase() ?? "";
                let postauthID = td.querySelector('span.uid')?.textContent?.match(/\d+/) ?? ["0"];
                if ((!thread.highImgOnly || td.querySelector('img') !== null) && //filter images
                  (thread.highNames.length == 0 || thread.highNames.includes(postauthName)) && //filter poster name
                  (thread.highIDs.length == 0 || thread.highIDs.includes(postauthID[0])) && //filter poster ID
                  (thread.ignoreIDs.length == 0 || !thread.ignoreIDs.includes(postauthID[0])) && //filter ignore ID
                  (thread.ignoreNames.length == 0 || !thread.ignoreNames.includes(postauthName)) //filter ignore name
                )
                  ++newCount;
              }
            });
            //
            //update threads
            thread.newEntrCnt = newCount;
            thread.totalEntrCnt = replyElements.length;
            thread.label = doc.querySelector("div.postwidth span.filetitle")?.innerText.trim() ?? thread.label;
            thread.author = doc.querySelector("div.postwidth span.postername")?.innerText.trim() ?? thread.author;
            this.wdata.save();
            //
            //update list
            let entrNewCnt = document.querySelector(`div.BLICK_wrow[data-id='${id}'] div.BLICK_wnew a`);
            if (entrNewCnt !== null) {
              entrNewCnt.innerHTML = String(newCount);
              entrNewCnt.href = this.generateLink(thread, id, curReadInd);
            }
            let entrLabel = document.querySelector(`div.BLICK_wrow[data-id='${id}'] div.BLICK_wtitle a`);
            if (entrLabel !== null)
              entrLabel.innerText = thread.label;
            let entrAuthor = document.querySelector(`div.BLICK_wrow[data-id='${id}'] div.BLICK_wauthor`);
            if (entrAuthor !== null)
              entrAuthor.innerText = thread.author;
          }));
      });
      Promise.all(fetchPromise).then(() => {
        document.getElementById("BLICK_watch_update")?.style.removeProperty("filter");
      });
    }
    insertEntries() {
      if (this.wdata.data.threads.size == 0) {
        this.cont.replaceChildren("No watched threads yet!");
        return;
      }
      const fragment = document.createDocumentFragment();
      const groupedThreads = new Map(this.thList.map(section => [section, []]));
      this.wdata.data.threads.forEach((thread, id) => { groupedThreads.get(thread.section)?.push(id); });
      groupedThreads.forEach((threads, section) => {
        if (threads.length > 0) {
          let head = document.createElement("div");
          head.className = "BLICK_whead";
          head.innerText = section;
          fragment.appendChild(head);
        }
        threads.forEach(id => {
          const el = this.wdata.data.threads.get(id);
          const row = document.createElement('div');
          row.className = "BLICK_wrow";
          row.dataset.id = id;
          row.innerHTML = `
          <div class="BLICK_wcell BLICK_wtitle"><a target="_blank" href="https://questden.org/kusaba/${el.section}/res/${id}.html">${el.label}</a></div>
          <div class="BLICK_wcell BLICK_wauthor">${el.author}</div>
          <div class="BLICK_wcell BLICK_wnew"><a target="_blank" href="${this.generateLink(el, id, -1)}">${el.newEntrCnt ?? 0}</a></div>
          <div class="BLICK_wcell BLICK_wactions">
              <button data-role='remove' title='Remove watch'>🗑️</button>
              <button data-role='options' title='Adjust settings'>🛠️</button>
          </div>`;
          fragment.appendChild(row);
        });
      });
      this.cont.replaceChildren(fragment);
    }
  }
  //
  //
  let imgQ;
  (function (imgQ) {
    imgQ[imgQ["convert"] = 0] = "convert";
    imgQ[imgQ["fullview"] = 1] = "fullview";
    imgQ[imgQ["thumbnail"] = 2] = "thumbnail";
  })(imgQ || (imgQ = {}));
  let imgF;
  (function (imgF) {
    imgF[imgF["webp"] = 0] = "webp";
    imgF[imgF["jpeg"] = 1] = "jpeg";
    imgF[imgF["unchanged"] = 2] = "unchanged";
  })(imgF || (imgF = {}));
  //
  class FileExport {
    constructor() {
      this.title = "";
      this.author = "";
      this.entryList = [];
      this.insertHTML();
      this.insertStyle();
      this.filter = {
        onlyImg: true, onlyCheck: false, imgQuality: imgQ.convert, include: "", exclude: "", style: 0,
        imgSets: { maxWidth: 800, maxHeight: 800, imgFormat: imgF.webp }, toEpub: true
      };
    }
    ;
    insertHTML() {
      this.form = document.createElement("div");
      this.form.style.display = "none";
      this.form.id = "BLICK_epub_form_overlay";
      this.form.innerHTML = `
        <div id="BLICK_epub_form">
          <h2>Export Thread to EPUB</h2>
          <label for="BLICK_epub_title">Title:</label>
          <input type="text" id="BLICK_epub_title" class="BLICK_epub_input" placeholder="Enter title" required="">
          <label for="BLICK_epub_author">Author:</label>
          <input type="text" id="BLICK_epub_author" class="BLICK_epub_input" placeholder="Enter author" required="">
          <div id='BLICK_epub_checkmarks'>
            <label for="BLICK_epub_includeImages">
              <input type="checkbox" id="BLICK_epub_includeImages">
              Only Include Posts with Images
            </label>
            <label for="BLICK_epub_includeChecked">
              <input type="checkbox" id="BLICK_epub_includeChecked">
              Only Include checked Posts
            </label>
          </div>
          <label for="BLICK_epub_imageQuality">Image Quality</label>
          <select id="BLICK_epub_imageQuality" class="BLICK_epub_select">
            <option value="convert">Convert images</option>
            <option value="fullsize">Use fullsize images</option>
            <option value="thumbnail">Use thumbnails</option>
          </select>
          <label for="BLICK_epub_imageSetting">Image Settings</label>
          <div id='BLICK_epub_imageSetting'>
            <label for="BLICK_epub_imageFormat">Format</label>
            <select id="BLICK_epub_imageFormat" class="BLICK_epub_select" title='Target image format.\nWebp is compresses best and supports animations, but is not yet supported by Kindle.\nJPG compresses photos and gradients well and is widely supported, but removes animations.\nNot converting preserves the format, but gif-animations are replaced by their first frame.'>
              <option value="webp">webp</option>
              <option value="jpg">JPG</option>
              <option value="unchanged">Don't convert</option>
            </select>
            <label for="BLICK_epub_imageMaxW">Max width:</label>
            <input type="text" id="BLICK_epub_imageMaxW" class="BLICK_epub_input" placeholder="Max width" title="Max width in px. 0 to ignore.">
            <label for="BLICK_epub_imageMaxH">Max height:</label>
            <input type="text" id="BLICK_epub_imageMaxH" class="BLICK_epub_input" placeholder="Max height" title="Max height in px. 0 to ignore.">
          </div>
          <label for="BLICK_epub_includeAuthors">Filter by Author:</label>
          <input type="text" id="BLICK_epub_includeAuthors" class="BLICK_epub_input" placeholder="Author ID or name" style="">
          <button data-role="detect" class="BLICK_epub_button" title='Generates a list of Names and IDs of the authors of posts with images'>Detect</button>
          <label for="BLICK_epub_excludeAuthors">Exclude Authors:</label>
          <input type="text" id="BLICK_epub_excludeAuthors" class="BLICK_epub_input" placeholder="Authors to exclude">
          <label for="BLICK_epub_styleSelect">Select Style:</label>
          <select id="BLICK_epub_styleSelect" class="BLICK_epub_select">
            <option>Simple, full-width images</option>
            <option>Simple, floating images with max. width</option>
          </select>
          <div id="BLICK_epub_stat">0 Posts, 0 Images</div>
          <div class="BLICK_epub_buttons">
            <button data-role="export" class="BLICK_epub_button">Export</button>
            <button data-role="cancel" class="BLICK_epub_button">Cancel</button>
            <button data-role="select" class="BLICK_epub_button" title='highlights and checks all posts that will go into the epub file'>Highlight</button>
          </div>
        </div>`;
      document.body.appendChild(this.form);
      this.statDispEl = document.getElementById("BLICK_epub_stat");
      this.form.addEventListener("click", (ev) => {
        if (ev.target === this.form)
          return this.hideEPUBForm();
        if (!(ev.target instanceof HTMLButtonElement))
          return;
        switch (ev.target.dataset.role) {
          case "detect":
            this.detectPosterName();
            break;
          case "select":
            this.transferSettings();
            this.selectEntries();
            break;
          case "export":
            this.transferSettings();
            if (this.filter.toEpub)
              this.exportEpub();
            else
              this.exportCBZ();
            break;
          case "cancel":
            this.hideEPUBForm();
            break;
          default:
            break;
        }
      });
      this.form.addEventListener("change", (ev) => {
        this.transferSettings();
        this.filterEntries(document);
      }, true);
      document.getElementById("BLICK_epub_imageQuality")?.addEventListener("change", ev => {
        let target = ev.target;
        let visibl = target.selectedIndex == imgQ.convert;
        document.getElementById("BLICK_epub_imageSetting")?.style.setProperty("display", visibl ? "" : "none");
        document.querySelector("label[for='BLICK_epub_imageSetting']")?.style.setProperty("display", visibl ? "" : "none");
      });
      document.getElementById("delform")?.addEventListener("click", (ev) => {
        let target = ev.target;
        if (target.matches("input[type='checkbox'][name='post[]']"))
          if (target.checked)
            target.parentElement?.classList.add("BLICK_epub_selected");
          else
            target.parentElement?.classList.remove("BLICK_epub_selected");
      });
    }
    insertStyle() {
      const sty = document.createElement("style");
      sty.id = 'BLICK_epub_style';
      sty.innerHTML = `
          #BLICK_epub_form_overlay{position: fixed;top: 0;left: 0;width: 100vw;height: 100vh;background-color: rgba(0, 0, 0, 0.5);display: flex;justify-content: center;align-items: center;z-index: 1000;}
          #BLICK_epub_form {color: #333; display: grid; grid-template-columns: 130px 3fr 1fr; gap: 15px 5px; border-radius: 8px; box-shadow: 2px 2px 5px rgba(0, 0, 0, 0.2); padding: 20px; width: 600px; margin: auto; align-items: center; border: 1px solid #b0a394;  background-color: #fffff0;}
          #BLICK_epub_form h2 {grid-column: span 3; margin: 0; text-align: center; color: #4CAF50;}
          #BLICK_epub_form .BLICK_epub_input,.BLICK_epub_select {padding: 10px; border: 1px solid #ccc; border-radius: 4px; box-sizing: border-box; grid-column: span 2;}
          #BLICK_epub_form .BLICK_epub_buttons {grid-column: span 3; display: flex; justify-content: space-between; margin-top: 10px;}
          #BLICK_epub_form .BLICK_epub_button {color: white; border: none; border-radius: 4px; padding: 10px 15px; cursor: pointer; transition: background-color 0.3s; flex: 1; margin: 0 5px;}
          #BLICK_epub_form .BLICK_epub_button:hover {filter: brightness(120%);}
          #BLICK_epub_form .BLICK_epub_button[data-role='export']{background-color:#3ba44b}
          #BLICK_epub_form .BLICK_epub_button[data-role='cancel']{background-color:#bb5546}
          #BLICK_epub_form .BLICK_epub_button[data-role='select']{background-color:#dd9243}
          #BLICK_epub_form .BLICK_epub_button[data-role='detect']{background-color:#dd9243;}
          #BLICK_epub_form #BLICK_epub_includeAuthors {grid-column: span 1;}
          #BLICK_epub_form label,#BLICK_epub_form input[type='checkbox']  {cursor: pointer;}
          #BLICK_epub_stat {grid-column: span 3;text-align: center;}
          .BLICK_epub_selected{background-color: #f005;}
          #BLICK_epub_imageSetting {display: flex;grid-column: span 2;align-items: center; justify-content: space-between;}
          #BLICK_epub_imageSetting label{text-align:center;}
          #BLICK_epub_imageFormat {width: 68px;}
          #BLICK_epub_imageMaxW, #BLICK_epub_imageMaxH {width: 60px;}
          #BLICK_epub_checkmarks{grid-column: 2 /span 2;display:flex;justify-content: space-between;}
          `;
      document.head.appendChild(sty);
    }
    showExportForm(mode = "epub") {
      if (this.form == null)
        return;
      this.form.style.display = "flex";
      //
      if (mode == "cbz") {
        this.form.querySelector("h2").innerHTML = "Export Thread to CBZ";
        this.form.querySelector("#BLICK_epub_styleSelect").style.display = "none";
        this.form.querySelector("label[for='BLICK_epub_styleSelect']").style.display = "none";
        this.form.querySelector("label[for='BLICK_epub_includeImages']").style.display = "none";
        this.filter.toEpub = false;
      }
      else if (mode = "epub") {
        this.form.querySelector("h2").innerHTML = "Export Thread to EPUB";
        this.form.querySelector("#BLICK_epub_styleSelect").style.display = "";
        this.form.querySelector("label[for='BLICK_epub_styleSelect']").style.display = "";
        this.form.querySelector("label[for='BLICK_epub_includeImages']").style.display = "";
        this.filter.toEpub = true;
      }
      else {
        alert("questden_blick2 error: unknown export form mode");
        throw new Error("questden_blick2 error: unknown export form mode: " + mode);
      }
      //	
      document.getElementById("BLICK_epub_includeImages").checked = this.filter.onlyImg;
      document.getElementById("BLICK_epub_includeChecked").checked = this.filter.onlyCheck;
      document.getElementById("BLICK_epub_includeAuthors").value = this.filter.include;
      document.getElementById("BLICK_epub_excludeAuthors").value = this.filter.exclude;
      document.getElementById("BLICK_epub_styleSelect").selectedIndex = this.filter.style;
      document.getElementById("BLICK_epub_imageQuality").selectedIndex = this.filter.imgQuality;
      document.getElementById("BLICK_epub_imageFormat").selectedIndex = this.filter.imgSets.imgFormat;
      document.getElementById("BLICK_epub_imageMaxW").value = String(this.filter.imgSets.maxWidth);
      document.getElementById("BLICK_epub_imageMaxH").value = String(this.filter.imgSets.maxHeight);
      //
      this.title = document.querySelector("div.postwidth span.filetitle")?.innerText ?? "Default";
      document.getElementById("BLICK_epub_title").value = this.title;
      this.author = document.querySelector("div.postwidth span.postername")?.innerText ?? "Bob";
      document.getElementById("BLICK_epub_author").value = this.author;
      this.filterEntries(document);
    }
    hideEPUBForm() { this.form.style.display = "none"; }
    transferSettings() {
      this.filter.onlyCheck = document.getElementById("BLICK_epub_includeChecked").checked;
      this.filter.onlyImg = document.getElementById("BLICK_epub_includeImages").checked;
      this.filter.include = document.getElementById("BLICK_epub_includeAuthors").value;
      this.filter.exclude = document.getElementById("BLICK_epub_excludeAuthors").value;
      this.filter.style = document.getElementById("BLICK_epub_styleSelect").selectedIndex;
      this.filter.imgQuality = document.getElementById("BLICK_epub_imageQuality").selectedIndex;
      //
      this.filter.imgSets.imgFormat = document.getElementById("BLICK_epub_imageFormat").selectedIndex;
      this.filter.imgSets.maxWidth = parseInt(document.getElementById("BLICK_epub_imageMaxW").value);
      this.filter.imgSets.maxHeight = parseInt(document.getElementById("BLICK_epub_imageMaxH").value);
      //
      this.title = document.getElementById("BLICK_epub_title").value ?? "Default";
      this.author = document.getElementById("BLICK_epub_author").value ?? "Bob";
    }
    filterEntries(cont) {
      this.entryList = [];
      let imgCnt = 0;
      cont.querySelectorAll("div.postwidth").forEach(el => {
        el.parentElement.classList.remove("BLICK_epub_selected");
        //
        let authName = el.querySelector("span.postername")?.innerText ?? "<noname>";
        let authID = el.querySelector("span.uid")?.innerText.substring(4) ?? "<noid>";
        let hasImg = el.querySelector("img.thumb") !== null;
        let elChecked = el.querySelector("input[type='checkbox'][name='post[]']")?.checked ?? false;
        //
        if (this.filter.onlyCheck === true && !elChecked)
          return;
        if (this.filter.onlyImg === true && hasImg === false)
          return;
        if (this.filter.include !== "" && (!this.filter.include.includes(authName) && !this.filter.include.includes(authID)))
          return;
        if (this.filter.exclude !== "" && (this.filter.exclude.includes(authName) || this.filter.exclude.includes(authID)))
          return;
        //
        if (hasImg)
          ++imgCnt;
        this.entryList.push(el);
      });
      document.getElementById("BLICK_epub_stat").innerHTML = `${this.entryList.length} Posts, ${imgCnt} Images`;
    }
    detectPosterName() {
      if (this.form === null)
        return;
      const authNam = new Set(), authID = new Set();
      document.querySelectorAll("img.thumb").forEach(el => {
        const par = el.closest("div.postwidth");
        if (par === null)
          return;
        authNam.add(par.querySelector("span.postername")?.innerText);
        authID.add(par.querySelector("span.uid")?.innerText.substring(4));
      });
      const resStr = [...authNam, ...authID].join(", ");
      document.getElementById("BLICK_epub_includeAuthors").value = resStr;
    }
    selectEntries() {
      this.filterEntries(document);
      document.querySelectorAll("input[type='checkbox'][name='post[]']").forEach(el => {
        el.checked = false;
        el.parentElement.classList.remove("BLICK_epub_selected");
      });
      this.entryList.forEach(el => {
        let cb = el.querySelector("input[type='checkbox'][name='post[]']");
        cb.checked = true;
        el.parentElement.classList.add("BLICK_epub_selected");
      });
    }
    async fetchImages(imgUrls) {
      let imgcnt = 0;
      const requests = imgUrls.map(async (url) => {
        try {
          const response = await fetch(url);
          ++imgcnt;
          if (!response.ok) {
            throw new Error("Failed to fetch image");
          }
          const blob = await response.blob();
          this.statDispEl.innerHTML = `Downloading Image: ${imgcnt}/${imgUrls.length}`;
          const name = url.split("/").pop() || "unknown";
          //
          let ret = { name, img: blob };
          if (this.filter.imgQuality === imgQ.convert)
            ret = await this.processImage(blob, name, this.filter.imgSets.maxWidth, this.filter.imgSets.maxHeight);
          return ret;
        }
        catch (ex) {
          console.log("error in fetching images!", ex, url);
          throw ex;
        }
      });
      return Promise.all(requests);
    }
    async processImage(blob, name, maxWidth = 800, maxHeight = 800) {
      const img = new Image();
      return new Promise((resolve, reject) => {
        img.onload = () => {
          const canvas = document.createElement("canvas");
          let { width, height } = img;
          if ((maxWidth > 0 && width > maxWidth) || (maxHeight > 0 && height > maxHeight)) {
            const scaleFactor = Math.min(maxWidth / width, maxHeight / height);
            width = width * scaleFactor;
            height = height * scaleFactor;
          }
          //
          canvas.width = width;
          canvas.height = height;
          const ctx = canvas.getContext("2d");
          if (ctx == null)
            return;
          ctx.drawImage(img, 0, 0, width, height);
          //
          const fileType = name.split('.')?.pop()?.toLowerCase() ?? null;
          const formatMap = { webp: 'image/webp', jpeg: 'image/jpeg', jpg: 'image/jpeg', png: 'image/png', gif: 'image/gif' };
          const origFormat = fileType !== null ? formatMap[fileType] : formatMap.jpeg;
          let outputFormat = origFormat;
          if (this.filter.imgSets.imgFormat === imgF.jpeg)
            outputFormat = formatMap.jpeg;
          else if (this.filter.imgSets.imgFormat === imgF.webp)
            outputFormat = formatMap.webp;
          canvas.toBlob((processedBlob) => {
            if (processedBlob !== null)
              resolve({ name, img: processedBlob });
            else
              reject();
          }, outputFormat);
        };
        img.onerror = reject;
        img.src = URL.createObjectURL(blob);
      });
    }
    exportCBZ() {
      this.filter.onlyImg = true;
      this.filterEntries(document);
      const imgs = new Set();
      this.entryList.forEach((el, ind, arr) => {
        this.statDispEl.innerHTML = `Processing Post: ${ind}/${this.entryList.length}%`;
        let cont = el.parentElement;
        const imgEl = cont?.querySelector("img.thumb");
        let imgName = "";
        if (imgEl !== null) {
          let imgUrl = imgEl.src;
          if (imgUrl.includes("spoiler.png"))
            imgUrl = imgEl.closest("a")?.href.replace(".", "s.").replace("src", "thumb") ?? ""; //thumbnail-link from full-link
          if (this.filter.imgQuality !== imgQ.thumbnail) { //all thumbnail. convert uses fullview as base
            imgUrl = imgUrl.replace("s.", ".").replace("thumb", "src"); //full-link from thumbnail-link 
          }
          imgName = imgUrl.split("/").pop() ?? "";
          if (imgName !== "." && imgName !== "")
            imgs.add(imgUrl); //exclude spoiler image without image. link ends then wiht "/."
        }
      });
      //
      const zip = new JSZip();
      //
      this.fetchImages([...imgs]).then((imageBlob) => {
        let imgCnt = 0;
        imageBlob.forEach(blob => {
          ++imgCnt;
          zip.file(`${String(imgCnt).padStart(4, '0')}_${blob.name}`, blob.img, { binary: true });
          this.statDispEl.innerHTML = `Adding Image: ${imgCnt}/${imageBlob.length}`;
        });
      }).then(() => {
        return zip.generateAsync({
          type: "blob",
          compression: "DEFLATE" //, 
          // compressionOptions: { level: 1 }
        }, (metadata) => {
          this.statDispEl.innerHTML = `Compressing: ${metadata.percent.toFixed(2)}%`;
        });
      }).then(blob => {
        const link = document.createElement("a");
        link.href = URL.createObjectURL(blob);
        link.download = `[${this.author}]_${this.title}.cbz`;
        link.click();
        this.statDispEl.innerHTML = `Downloading.`;
      });
    }
    exportEpub() {
      this.filterEntries(document);
      const imgs = new Set();
      const cont = this.entryList.map((el, ind, arr) => {
        this.statDispEl.innerHTML = `Processing Post: ${ind}/${this.entryList.length}%`;
        let cont = el.parentElement;
        //
        const author = cont?.querySelector("span.postername")?.innerText ?? "Bob";
        const title = cont?.querySelector("span.filetitle")?.innerText ?? "";
        const id = cont?.querySelector("input[type='checkbox'][name='post[]']")?.value ?? "";
        const date = cont?.querySelector("span.postername")?.nextSibling?.textContent?.trim() ?? "";
        //
        let msg = cont?.querySelector("blockquote")?.innerHTML ?? ""; //message body
        msg = msg.replace(/src="\/kusaba\/.*?\/thumb\//gi, "src=\"images/") //icons/thumbnails
          .replace(/href="\/kusaba\/.*?\/res\/.*?#/gi, "href=\"#") //references
          .replace(/onmouseover=".*?"/gi, "") //remove onmouseover
          .replace(/onmouseout=".*?"/gi, "") //remove onmouseout
          .replace(/<a .*?>\s*&gt;&gt;(\d+)/gi, "<a data-refid='$1' onmouseenter='hoverPrevEnter(event)' onmouseleave='hoverPrevLeave(event)' href='#$1'>&gt;&gt;$1"). //adjust reflinks
          replace('<div style="display:inline-block; width:400px;"></div><br>', ""); //remove leading spacers
        cont?.querySelector("blockquote")?.querySelectorAll("img").forEach(el => { imgs.add(el.src); }); //download msg images/icons
        //
        const imgEl = cont?.querySelector("img.thumb");
        let imgName = "";
        if (imgEl !== null) {
          let imgUrl = imgEl.src;
          if (imgUrl.includes("spoiler.png"))
            imgUrl = imgEl.closest("a")?.href.replace(".", "s.").replace("src", "thumb") ?? ""; //thumbnail-link from full-link
          if (this.filter.imgQuality !== imgQ.thumbnail) { //all thumbnail. convert uses fullview as base
            imgUrl = imgUrl.replace("s.", ".").replace("thumb", "src"); //full-link from thumbnail-link 
          }
          imgName = imgUrl.split("/").pop() ?? "";
          if (imgName !== "." && imgName !== "")
            imgs.add(imgUrl); //exclude spoiler image without image. link ends then wiht "/."
        }
        //
        let tex = `<div id='${id}' class='post'>`;
        if (imgName !== "")
          tex += `<img class='thumb' src='images/${imgName}' />`;
        tex += `<div class='postHead'>
          <span class='post_author'>${author}</span>
          <span class='post_title'>${title}</span>
          <span class='post_date'>${date}</span>
          <span class='post_id'>#${id}</span>
        </div><div class='postBody'>${msg}</div></div>`;
        return tex;
      }).join("");
      //
      const zip = new JSZip();
      zip.file("mimetype", "application/epub+zip", { compression: "STORE" });
      zip.file("META-INF/container.xml", `
        <?xml version="1.0" encoding="UTF-8"?>
        <container version="1.0" xmlns="urn:oasis:names:tc:opendocument:xmlns:container">
            <rootfiles>
                <rootfile full-path="OEBPS/content.opf" media-type="application/oebps-package+xml"/>
            </rootfiles>
        </container>
      `);
      zip.file("OEBPS/content.opf", `
        <?xml version="1.0" encoding="UTF-8"?>
        <package xmlns="http://www.idpf.org/2007/opf" unique-identifier="BookId" version="2.0">
            <metadata xmlns:dc="http://purl.org/dc/elements/1.1/">
                <dc:title>${this.title}</dc:title>
                <dc:creator>${this.author}</dc:creator>
                <dc:language>en</dc:language>
            </metadata>
            <manifest>
                <item id="content" href="content.html" media-type="application/xhtml+xml"/>
                <item href="style.css" id="css" media-type="text/css" />
            </manifest>
            <spine>
                <itemref idref="content"/>
            </spine>
        </package>
      `);
      switch (this.filter.style) {
        default:
        case 0:
          zip.file("OEBPS/style.css", `
            div.post{margin-bottom: 25px;border-bottom: 1px dashed #ccc;padding: 10px;}
            img.thumb{width:90%;display: block;margin:auto;}
            span.post_title {color: red;font-weight: bold;}
            span.post_author {color: green;font-weight: bold;}
            div.postHead{display:flex;display: flex;justify-content: space-between;margin-bottom:15px;flex-wrap: wrap;}
            #hover-content {background-color: #f9f9f9;padding: 10px;border: 1px solid #ccc;box-shadow: 0px 0px 10px rgba(0,0,0,0.1);z-index: 1000;display: none;}
            span.unkfunc{color: #789922;}
          `);
          break;
        case 1:
          zip.file("OEBPS/style.css", `
          div.post{margin-bottom: 25px;border-bottom: 1px dashed #ccc;padding: 10px;}
          img.thumb{display: block;float:left;width:100%;max-width:300px;margin-right:10px;}
          div.post::after {content: "";display: table;clear: both;}
          span.post_title {color: red;font-weight: bold;}
          span.post_author {color: green;font-weight: bold;}
          div.postHead{display:flex;justify-content: space-between;margin-bottom:15px;min-width:200px;flex-wrap: wrap;}
          #hover-content {background-color: #f9f9f9;padding: 10px;border: 1px solid #ccc;box-shadow: 0px 0px 10px rgba(0,0,0,0.1);z-index: 1000;display: none;}
          span.unkfunc{color: #789922;}
        `);
          break;
      }
      zip.file("OEBPS/content.html", `
        <?xml version="1.0" encoding="UTF-8"?>
        <!DOCTYPE html>
        <html xmlns="http://www.w3.org/1999/xhtml">
            <head>
              <title>${this.title}</title>
              <link href="style.css" rel="stylesheet" type="text/css" />
              <script>
                function hoverPrevEnter(e){
                  const hoverContent = document.getElementById('hover-content');
                  hoverContent.style.left = e.pageX + 15 + 'px'; // 15px Offset from the cursor
                  hoverContent.style.top = e.pageY + 15 + 'px';									const targetId = e.target.dataset.refid;
                  const targetElement = document.getElementById(targetId);
                  if (targetElement) {
                      hoverContent.innerHTML = targetElement.innerHTML;
                      hoverContent.style.display = 'block';
                  }
                };
                function hoverPrevLeave(e){
                  const hoverContent = document.getElementById('hover-content');
                  hoverContent.style.display = 'none';
                  hoverContent.innerHTML = '';
                }
              </script>
            </head>						
            <body>
              <header>${this.title}</header>
              <div id='questauthor'>By ${this.author}</div>
              ${cont}
            <div id="hover-content" style="position: absolute; display: none;"></div>
            </body>
        </html>
      `);
      this.fetchImages([...imgs]).then((imageBlob) => {
        let imgCnt = 0;
        imageBlob.forEach(blob => {
          ++imgCnt;
          zip.file(`OEBPS/images/${blob.name}`, blob.img, { binary: true }); //
          this.statDispEl.innerHTML = `Adding Image: ${imgCnt}/${imageBlob.length}`;
        });
      }).then(() => {
        return zip.generateAsync({
          type: "blob",
          compression: "DEFLATE" //, 
          // compressionOptions: { level: 1 }
        }, (metadata) => {
          this.statDispEl.innerHTML = `Compressing: ${metadata.percent.toFixed(2)}%`;
        });
      }).then(blob => {
        const link = document.createElement("a");
        link.href = URL.createObjectURL(blob);
        link.download = `[${this.author}]_${this.title}.epub`;
        link.click();
        this.statDispEl.innerHTML = `Downloading.`;
      });
    }
  }
  class DOMManip {
    //
    constructor(set) {
      this.setting = set;
      //
      this.EvImgHoverIn = this.EvImgHoverIn.bind(this);
      this.EvImgHoverOut = this.EvImgHoverOut.bind(this);
      this.repform = new ReplyForm();
      this.watchbar = null;
      //
      this.insertStyleEl();
      this.insertHoverImg();
    }
    ;
    //
    insertStyleEl() {
      this.styEl = document.createElement("style");
      this.styEl.id = 'BLICK_manipStyle';
      document.head.appendChild(this.styEl);
      //
      this.styles = { fontsize: "", paragraphMargin: "", invertCol: "", replyForm: "" };
    }
    insertHoverImg() {
      this.hoverImgCont = document.createElement("div");
      this.hoverImgCont.id = "BLICK_imgbox";
      //
      this.wartImg = document.createElement("img");
      this.wartImg.src = imgRes.wartbild;
      this.wartImg.className = "loading";
      this.hoverImgCont.appendChild(this.wartImg);
      //
      this.fullImg = document.createElement("img");
      this.fullImg.src = imgRes.wartbild;
      this.fullImg.className = "imgEl";
      this.hoverImgCont.appendChild(this.fullImg);
      //
      document.body.appendChild(this.hoverImgCont);
      //
      this.hoverImgCont.addEventListener('mouseenter', this.EvImgHoverIn);
      this.hoverImgCont.addEventListener('mouseleave', this.EvImgHoverOut);
    }
    updateStyle() {
      this.styEl.innerHTML = Object.values(this.styles).join("\n");
    }
    initStyles() {
      this.setFontsize();
      this.setMargin();
      this.setInvert();
      this.setReplyForm();
      this.setWatchbar();
      this.setHoverImg();
      this.updateStyle();
    }
    //
    setFontsize(update = false) {
      if (!this.setting.data.changeFont) {
        this.styles.fontsize = "";
      }
      else {
        const ffam = this.setting.fontTypeList[this.setting.data.fonttype];
        const fsize = this.setting.data.fontsize + "pt";
        const templ = `#delform .bbcode_quote{line-height:1.5em;font-family:${ffam};font-size:${fsize};}
          #delform h2.title {font-family:${ffam};font-weight:bold;font-size: ${fsize};}
          #delform {line-height:1.5em;font-family:${ffam};font-size:${fsize};margin-right: 7%;margin-top: 15px;margin-bottom: 25px;text-align: left;letter-spacing: 1px;color: #003;}`;
        this.styles.fontsize = templ;
      }
      if (update)
        this.updateStyle();
    }
    setMargin(update = false) {
      if (!this.setting.data.paragraphMargin) {
        this.styles.paragraphMargin = "";
      }
      else {
        const pg = window.location.href.search(/\/[\d\+]+\.html/) == -1;
        const inv = !this.setting.data.invertCol;
        //
        const templ = `#delform>br,#delform>hr {display:none;}
          #delform${pg ? ">div" : ""}{box-shadow:0px 0px 5px 3px #${inv ? "000" : "fff"},5px 5px 15px 5px #${!inv ? "000" : "fff"},inset 0px 0px 5px #${!inv ? "000" : "fff"};margin-bottom:30px;	border-radius:5px;	border: 2px groove #000;}
          #delform{width: 100%;}
          .thumb{box-shadow:0px 0px 15px ${inv ? "#fff" : "#000"};	border-radius:10px;}
          .reply{width: 100%;}
          .userdelete{clear:both;}
          #delform>div{padding: 10px;}
          #delform > div::after {content: "";display: inline-block;clear: both;}
          #delform > div > div[id] {clear: both;}
          #delform>span {	margin-top:10px;}
          #delform img.inlineimg {vertical-align: middle!important;	margin: 0 4px;}
      `; //width: 98%;display:table;
        this.styles.paragraphMargin = templ;
      }
      if (update)
        this.updateStyle();
    }
    setInvert(update = false) {
      if (!this.setting.data.invertCol) {
        this.styles.invertCol = "";
      }
      else {
        const pg = window.location.href.search(/\/[\d\+]+\.html/) == -1;
        const templ = `html, body{background-color:#002!important;}
          #delform a {color: #BB5555;font-size: 14px;font-style: italic;text-decoration: underline;}
          #delform, #delform .reply, #delform blockquote{background-color:#001;color: #FFD;}
          #delform span.spoiler{text-shadow:none;}
          #delform .highlight{background-color:#003;}
          ${this.setting.data.paragraphMargin ? ".thumb{	box-shadow:0px 0px 15px #fff;	border-radius:10px;}" : ""}
          "#delform h2.title{color:#fdd}
          ${this.setting.data.paragraphMargin ? ".thumb{	box-shadow:0px 0px 15px #fff;	border-radius:10px;}" : ""}
          ${this.setting.data.paragraphMargin ? ((pg ? "#delform>div{" : "#delform>div{") + "box-shadow:0px 0px 5px 3px #fff,5px 5px 15px 5px #000000,inset 0px 0px 5px #000;}") : ""}
          #delform {color:#ffd}`;
        this.styles.invertCol = templ;
      }
      if (update)
        this.updateStyle();
    }
    setReplyForm(update = false) {
      if (!this.setting.data.replyForm) {
        this.styles.replyForm = "";
        this.repform.removeBar();
        document.querySelector("#BLICK_previewbut")?.remove();
        document.querySelector("#BLICK_cb_preview")?.remove();
      }
      else {
        this.styles.replyForm = `
        #BLICK_previewbut{margin-right:0px!important;}/*workaround questreader*/
        td.postblock + td { display: grid; grid-template-columns: 1fr 30px auto!important; }/*workaround questreader*/
        .postarea table tr td:nth-of-type(2) {display: flex;gap: 5px;align-items: center;}
        .postarea table tr td a.rules{width:auto!important;}
        .postarea table tr td input[type="text"] {flex: 1;}
        .postarea table tr td textarea{width: 100%;  box-sizing: border-box;}
        input[name="postpassword"]{flex:0.5!important;}
        `;
        document.querySelectorAll(".postarea table tr td:nth-of-type(2)").forEach(el => {
          const prevEl = el.querySelector("input[type='text']") ?? null;
          if (prevEl === null)
            return;
          let tex = "";
          el.childNodes.forEach(node => {
            if (node.nodeType == Node.TEXT_NODE) {
              tex += node.textContent?.trim();
              el.removeChild(node);
            }
          });
          if (tex !== "") {
            tex = tex.replace(/\(|\)/g, "");
            prevEl.placeholder = tex;
            prevEl.title = tex;
          }
        });
        let pwbox = document.querySelector(`input[name="postpassword"]`);
        if (pwbox?.nextSibling === null)
          pwbox?.insertAdjacentHTML("afterend", "<span></span>"); //questReader workaround for .nextSibling.remove()
        this.repform.insertBar();
      }
      if (update)
        this.updateStyle();
    }
    setWatchbar() {
      if (this.setting.data.manageWatchlist) {
        if (this.watchbar === null)
          this.watchbar = new WatchBar(this.setting);
      }
      else {
        if (this.watchbar !== null) {
          this.watchbar.destroy();
          this.watchbar = null;
        }
      }
    }
    setHoverImg() {
      if (this.setting.data.imageHover) {
        document.querySelectorAll('img.thumb:not([BLICK_Hover])').forEach(img => {
          img.addEventListener('mouseover', this.EvImgHoverIn);
          img.addEventListener('mouseleave', this.EvImgHoverOut);
        });
      }
      else {
        document.querySelectorAll("img.thumb[BLICK_Hover]").forEach(img => {
          img.removeEventListener("mouseover", this.EvImgHoverIn);
          img.removeEventListener("mouseleave", this.EvImgHoverOut);
          img.removeAttribute("BLICK_Hover");
        });
      }
    }
    EvImgHoverIn(ev) {
      if (!(ev.target instanceof HTMLImageElement))
        return;
      //
      const linkEl = ev.target.parentNode?.parentNode;
      if (linkEl == null)
        return;
      const fullImageUrl = linkEl.getAttribute('href') ?? "";
      //
      this.wartImg.style.display = 'block';
      this.fullImg.style.display = 'none';
      this.hoverImgCont.style.display = 'block';
      //
      this.fullImg.src = fullImageUrl;
      this.fullImg.onload = () => {
        this.wartImg.style.display = 'none';
        this.fullImg.style.display = 'block';
      };
    }
    EvImgHoverOut(ev) {
      if (!this.hoverImgCont.contains(ev.relatedTarget)) {
        this.hoverImgCont.style.display = 'none';
      }
    }
  }
  //
  // start
  //
  function main() {
    //action starts after storage is loaded/sanitized
    delete Array.prototype.toJSON; //remove prototype.js prototype changes to stringify()
    const set = new Settings();
    const syncr = new Sync();
    const dom = new DOMManip(set);
    const sb = new Sidebar(dom, set, syncr);
    set.load().then(() => {
      sb.loadSettings();
      dom.initStyles();
    }).catch(ex => console.log("Blick error: critical initialization error", ex));
  }
  //
  //loading storage
  //
  main();
})();
//# sourceMappingURL=questden_blick2.user.js.map