LemmyTools

A small suite of tools to make Lemmy easier.

  1. // ==UserScript==
  2. // @name LemmyTools
  3. // @namespace https://thesimplecorner.org/c/lemmytools
  4. // @version 0.2.1.0
  5. // @description A small suite of tools to make Lemmy easier.
  6. // @author howdy@thesimplecorner.org
  7. // @author @cwagner@lemmy.cwagner.me
  8. // @grant none
  9. // @include https://*
  10. // ==/UserScript==
  11.  
  12. (function () {
  13. "use strict";
  14. if (!isLemmySite()) { //not lemmy
  15. return;
  16. }
  17. else //is Lemmy Do!
  18. {
  19. // ------------ EDIT THIS VARIABLE ---------------------
  20. const homeInstance = "";
  21. // -----------------------------------------------------
  22. // homeInstance - Fixes remote Instance home link. Example: const homeInstance = "https://lemmy.world";
  23. /* NOTHING NEEDS CHANGED BELOW */
  24. const LogDebug = 0;
  25. const LogInformation = 1;
  26. const LogImportant = 2;
  27. // Choose a log level for the console:
  28. const logLevel = LogDebug;
  29. // const logLevel = LogInformation;
  30. // const logLevel = LogImportant;
  31. // ------------ END EDIT AREA --------------------------
  32. //Nothing below needs editing.
  33. // -------------- VERSION -------------------
  34. const ltVer = "0.2.1.0";
  35. const ltTestedVer = "0.19.1";
  36. //--------------------------------------------
  37.  
  38. /* Globals */
  39.  
  40. const mobile = isltMobile();
  41. let remoteCommunityArray = [];
  42. let prevSearchCommsQueries = [];
  43. prevSearchCommsQueries.push("");
  44. let currentUrl = document.location.href;
  45.  
  46. /**
  47. * @type {Map<string, {Text: string, Color: string, Url?: string}>}
  48. */
  49. let userMap;
  50.  
  51. function isHomeInstanceSet(i2c) {
  52. return i2c.length > 3;
  53. }
  54.  
  55. function ltLog(msg, level) {
  56. level = level || LogImportant;
  57. if (level < logLevel) return;
  58. console.log("[LemmyTools]", msg);
  59. }
  60.  
  61.  
  62. function notHomeAndInCommunity(url) {
  63. return (
  64. url.includes(settings.instance) === false &&
  65. (url.includes("/c/") ||
  66. url.includes("/post/") ||
  67. url.includes("/comment/") ||
  68. url.includes("/communities"))
  69. );
  70. }
  71.  
  72. function isltMobile() {
  73. if (/Android|iPhone/i.test(navigator.userAgent)) {
  74. ltLog("is mobile!");
  75. return true;
  76. } else {
  77. ltLog("is desktop!");
  78. return false;
  79. }
  80. }
  81.  
  82. //Remote Instance
  83. function update(comm, commName, subString) {
  84. try {
  85. if (comm) {
  86. const browsedComm = `<li><h5>${comm}</h5></li>
  87. <li>
  88. <a href='${homeInstance}/c/${commName}' target='_blank'><button type="button" class='ltbutton'>Browse/Sub on Home Instance</button></a>
  89. </li>`;
  90. remoteCommunityArray.push(browsedComm);
  91. }
  92. } catch {}
  93.  
  94. return remoteCommunityArray;
  95. }
  96.  
  97. //Searches communityArray for results in LemmyTools Sidebar.
  98. function searchComms(query, full) {
  99. ltLog(`commsearch evt searchinput${query}${commsAreaStatic}`, LogDebug);
  100. const url = window.location.href;
  101. query = query || "";
  102. query = query.toLowerCase();
  103. if ((query == "-f") && (prevSearchCommsQueries.length < 2)) {
  104. const commsCount = localStorage.getItem("commsCount");
  105. if (commsCount == null || commsCount == 0 || full.length < 1) {
  106. commsAreaStatic[0].innerHTML = `<hr /><b>Welcome to LemmyTools! Ver ${ltVer}!</b><br /><br />
  107. First time? Set your lemmy homeinstance in the option page and in the UserScript.<br />
  108. No communities? Login to lemmy and reload page.`;
  109. } else {
  110. commsAreaStatic[0].innerHTML = `Communities: ${commsCount} - <hr />${full}`;
  111. }
  112. } else {
  113. //This searches the pushed communityArray with the query, saves it to a array, removes any duplicate values, sorts and then pushes to the commupdate function.
  114. commsAreaStatic[0].innerHTML = full;
  115. //if searchInput query, store it for use on another page
  116. if (query.length > 2)
  117. {
  118. prevSearchCommsQueries.push(query);
  119. localStorage.setItem("prevSearchCommsQueries", prevSearchCommsQueries);
  120. }
  121. //ltLog(`Searching for:${query}`, LogDebug);
  122. const children = commsAreaStatic[0].getElementsByTagName("li");
  123. //ltLog(`Children found: ${children.length}`, LogDebug);
  124. let data = [""];
  125. let found;
  126. for (let i = 0; i < children.length; i++) {
  127. if (children[i].innerHTML.toLowerCase().indexOf(query) !== -1) {
  128. found = children[i].innerHTML + "<br />";
  129. //ltLog(`Found: ${found}`, LogDebug);
  130. data.push(found);
  131. }
  132. }
  133. const resultSet = [...new Set(data)];
  134. resultSet.sort();
  135. if (currentUrl.indexOf(query) !== -1)
  136. {
  137. commupdate(url, resultSet, query);
  138. }
  139. else
  140. {
  141. commupdate(url, resultSet, query);
  142. }
  143. }
  144. }
  145.  
  146. function commupdate(page, data, query) {
  147. ltLog("LTbar Update");
  148. let count = -1;
  149. data.forEach((_) => count++);
  150. data = data.join("");
  151. for (let i = 0; i < commsAreaSearch.length; i++)
  152. {
  153. commsAreaSearch[i].innerHTML = `Communities: ${count}<hr /> ${data}`;
  154. }
  155.  
  156. for (let i = 0; i < commsAreaStatic.length; i++)
  157. {
  158. commsAreaStatic[i].innerHTML = `Communities: ${count}<hr /> ${data}`;
  159. }
  160. if (query.length > 2)
  161. {
  162. searchInput.value = query;
  163. }
  164. }
  165. const optionsKey = "LemmyToolsOptions";
  166.  
  167. function getSettingsFromLocalStorage() {
  168. try {
  169. return JSON.parse(localStorage.getItem(optionsKey) || "{}");
  170. } catch (_) {
  171. return {};
  172. }
  173. }
  174.  
  175. /**
  176. * Gets all currently tagged users
  177. *
  178. * @return {Map<string, {Text: string, Color: string, Url: string}>} A Map of tagged users
  179. */
  180. function getUserCommentsFromLocalStorage() {
  181. try {
  182. const notes = localStorage.getItem(optionsKey + "-userNotes") || "[]";
  183. return new Map(JSON.parse(notes));
  184. } catch (e) {
  185. console.error(e);
  186. return new Map();
  187. }
  188. }
  189.  
  190. function saveUserCommentsToLocalStorage() {
  191. localStorage.setItem(
  192. optionsKey + "-userNotes",
  193. JSON.stringify(Array.from(userMap))
  194. );
  195. }
  196.  
  197. function options(open) {
  198. const odiv = document.getElementById("ltOptions");
  199. ltLog(`Options Functions: ${open}`);
  200. let userOptions = {};
  201. if (open === 1) {
  202. odiv.style.display = "block";
  203. } else if (open === 2) {
  204. //First run set defaults or pull from localstorage.
  205. const mobile = isltMobile();
  206. userOptions = Object.assign(
  207. {},
  208. {
  209. commposSide: mobile ? "top" : "top",
  210. reverseSide: mobile ? "left" : "right",
  211. instance: homeInstance || window.location.origin,
  212. commposVertical: 0,
  213. expandImages: true,
  214. hideSideBar: false,
  215. expandImagesize: mobile ? 100 : 50,
  216. hoverCheck: false,
  217. unblurNSFW: false,
  218. widthPixels: false,
  219. blockContent: false,
  220. blockFilters: "",
  221. expandImageSpeed: 0.5,
  222. showAllImages: false,
  223. hideShowAllImagesButton: false,
  224. linksInNewTab:false,
  225. },
  226. getSettingsFromLocalStorage()
  227. );
  228. localStorage.setItem(optionsKey, JSON.stringify(userOptions));
  229. userMap = getUserCommentsFromLocalStorage();
  230. } else if (open === 3) {
  231. //save button
  232. odiv.style.display = "none";
  233. localStorage.setItem("currentBlockCount", 0);
  234.  
  235. userOptions.commposSide =
  236. document.getElementById("option_commposSide").value;
  237. userOptions.instance = document.getElementById(
  238. "option_homeInstance"
  239. ).value;
  240. userOptions.commposVertical = parseInt(
  241. document.getElementById("option_commposVertical").value
  242. );
  243. userOptions.expandImages = document.getElementById(
  244. "option_expandImages"
  245. ).checked;
  246. userOptions.expandImagesize = parseInt(
  247. document.getElementById("option_expandImagesize").value,
  248. 10
  249. );
  250. userOptions.expandImageSpeed = parseFloat(
  251. document.getElementById("option_expandImageSpeed").value
  252. );
  253. userOptions.hideSideBar =
  254. document.getElementById("option_hideSideBar").checked;
  255. userOptions.hoverCheck =
  256. document.getElementById("option_hoverCheck").checked;
  257. userOptions.unblurNSFW =
  258. document.getElementById("option_unblurNSFW").checked;
  259. userOptions.blockContent =
  260. document.getElementById("option_blockContent").checked;
  261. userOptions.blockFilters =
  262. document.getElementById("option_blockFilters").value.split(",");
  263. userOptions.linksInNewTab = document.getElementById(
  264. "option_linksInNewTab"
  265. ).checked;
  266. userOptions.widthPixels =
  267. document.getElementById("option_widthPixels").checked;
  268. userOptions.showAllImages = document.getElementById(
  269. "option_showAllImages"
  270. ).checked;
  271. userOptions.hideShowAllImagesButton = document.getElementById(
  272. "option_hideShowAllImagesButton"
  273. ).checked;
  274.  
  275. if (userOptions.commposVertical > 85) {
  276. userOptions.commposVertical = 85;
  277. } else if (userOptions.commposVertical <= -1) {
  278. userOptions.commposVertical = 0;
  279. }
  280.  
  281. if (userOptions.expandImageSpeed > 1) {
  282. userOptions.expandImageSpeed = 1;
  283. } else if (userOptions.expandImageSpeed < 0) {
  284. userOptions.expandImageSpeed = 0;
  285. }
  286.  
  287. if (userOptions.commposSide === "left") {
  288. userOptions.reverseSide = "right";
  289. } else {
  290. userOptions.reverseSide = "left";
  291. }
  292.  
  293. localStorage.setItem(optionsKey, JSON.stringify(userOptions));
  294. location.reload(true);
  295. }
  296.  
  297. userOptions = getSettingsFromLocalStorage();
  298. ltLog(`Settings ${JSON.stringify(userOptions)}`);
  299. return userOptions;
  300. }
  301.  
  302. //Used for offset removal
  303. function removeClassByWildcard(divClass) {
  304. // If the class ends with a "*", then it matches all classes that start with the given class name.
  305. if (divClass.endsWith("*")) {
  306. divClass = divClass.replace("*", "");
  307. // Get all elements with the given class name.
  308. const elements = document.getElementsByTagName("div");
  309. const re = new RegExp("(^|s)" + divClass + "(s|$)");
  310. const result = [];
  311. let className = "";
  312.  
  313. for (let i = 0; i < elements.length; i++) {
  314. if (re.test(elements[i].className)) {
  315. console.log("Match: " + elements[i]);
  316. result.push(elements[i]);
  317. for (let y = 0; y < elements[i].classList.length; y++) {
  318. if (elements[i].classList[y].indexOf(divClass) !== -1) {
  319. className = elements[i].classList[y];
  320. console.log(className);
  321. }
  322. }
  323. }
  324. }
  325. // Remove the class from all elements.
  326. for (let i = 0; i < result.length; i++) {
  327. result[i].classList.remove(className);
  328. }
  329. } else {
  330. // Otherwise, the class must match exactly.
  331. const elements = document.querySelectorAll("[class=" + divClass + "]");
  332.  
  333. // Remove the class from all elements.
  334. for (let i = 0; i < elements.length; i++) {
  335. elements[i].classList.remove(divClass);
  336. }
  337. }
  338. }
  339.  
  340. const colors = new Map([
  341. ["transparent", "inherit"],
  342. ["tomato", "white"],
  343. ["darkorange", "white"],
  344. ["gold", "black"],
  345. ["yellowgreen", "white"],
  346. ["lightseagreen", "white"],
  347. ["teal", "white"],
  348. ["indianred", "black"],
  349. ["lightcoral", "black"],
  350. ["lightpink", "black"],
  351. ["mistyrose", "black"],
  352. ["purple", "white"],
  353. ["crimson", "white"],
  354. ["teal", "white"],
  355. ["darkslategray", "white"],
  356. ]);
  357.  
  358. function addNotesToLinks(userNodes) {
  359. const userNameRegex = /\/u\/(.+)$/;
  360. for (const node of userNodes) {
  361. let userName;
  362. try {
  363. userName = node.href.match(userNameRegex)[1];
  364. } catch (error) {
  365. console.error(error, node, node.href);
  366. }
  367. let alreadyHadSpan = false;
  368. let span;
  369. const existing = node.getElementsByClassName("userNote");
  370. if (existing.length === 1) {
  371. alreadyHadSpan = true;
  372. span = existing[0];
  373. } else span = document.createElement("span");
  374.  
  375. span.innerHTML = "🏷";
  376. if (userMap.has(userName)) {
  377. span.innerHTML = userMap.get(userName).Text;
  378. span.style.backgroundColor = userMap.get(userName).Color;
  379. span.style.color = colors.get(userMap.get(userName).Color);
  380. }
  381. span.classList.add("userNote", "badge");
  382. span.dataset.user = userName;
  383. if (!alreadyHadSpan) {
  384. span.addEventListener("click", tagClick);
  385. node.appendChild(span);
  386. }
  387. node.classList.add("hasNote");
  388. }
  389. }
  390.  
  391. function tagUsers() {
  392. setInterval(() => {
  393. const userNodes = document.querySelectorAll(
  394. ".person-listing:not(.hasNote)"
  395. );
  396. addNotesToLinks(userNodes);
  397. }, 500);
  398. }
  399.  
  400. function tagClick(clickEvent) {
  401. clickEvent.preventDefault();
  402. const userName = clickEvent.currentTarget.dataset.user;
  403. const dialog = document.createElement("dialog");
  404. let colorOptions = "";
  405. colors.forEach(
  406. (val, key) =>
  407. (colorOptions += `<option style="color: ${val}; background-color: ${key}" value="${key}">${key}</option>`)
  408. );
  409. dialog.classList.add("userTagger");
  410. dialog.addEventListener("click", (e) => {
  411. if (e.target === dialog) {
  412. closeDialog(dialog);
  413. }
  414. });
  415. dialog.innerHTML = `<div style="width: 350px;">
  416. <strong>${userName}</strong>
  417. <form id="userTag">
  418. <div>
  419. <label for="userTagText">Text</label>
  420. <input type="text" id="userTagText" value="">
  421. </div>
  422. <div>
  423. <label for="userTagColor">Color</label>
  424. <select id="userTagColor">
  425. ${colorOptions}
  426. </select>
  427. </div>
  428. <div>
  429. <button type="submit">✓ save tag</button>
  430. <button formmethod="dialog" cancel</button>
  431. </div>
  432. </form>
  433. </div>`;
  434.  
  435. const userTagText = dialog.querySelector("#userTagText");
  436. const userTagColor = dialog.querySelector("#userTagColor");
  437. if (userMap.has(userName)) {
  438. userTagText.value = userMap.get(userName).Text;
  439. userTagColor.value = userMap.get(userName).Color;
  440. }
  441. const form = dialog.querySelector("#userTag");
  442. form.addEventListener("submit", (e) => {
  443. e.preventDefault();
  444. if (e.submitter.formMethod === "dialog") {
  445. closeDialog(dialog);
  446. return;
  447. }
  448. if (userTagText.value.length) {
  449. if (!userMap.has(userName))
  450. userMap.set(userName, { Text: "", Color: "" });
  451. userMap.get(userName).Text = userTagText.value;
  452. userMap.get(userName).Color = userTagColor.value;
  453. } else {
  454. userMap.delete(userName);
  455. }
  456. addNotesToLinks(document.querySelectorAll(`[href="/u/${userName}"]`));
  457. saveUserCommentsToLocalStorage();
  458. closeDialog(dialog);
  459. });
  460. document.body.appendChild(dialog);
  461. dialog.showModal();
  462. }
  463. function closeDialog(dialog) {
  464. dialog.close();
  465. dialog.remove();
  466. }
  467. function ready(fn) {
  468. if (document.readyState !== "loading") {
  469. fn();
  470. } else {
  471. document.addEventListener("DOMContentLoaded", fn);
  472. }
  473. }
  474.  
  475. //Expand all images on page
  476. function allImages(show) {
  477. let clickableImages = document.getElementsByClassName(
  478. "thumbnail rounded overflow-hidden d-inline-block position-relative p-0 border-0 bg-transparent"
  479. );
  480.  
  481. //ltLog(clickableImages.length, LogDebug);
  482. //ltLog(clickableImages, LogDebug);
  483. if (show) {
  484. for (let i = 0; i < clickableImages.length; i++) {
  485. try {
  486. console.log(clickableImages[i]);
  487.  
  488. clickableImages[i].click();
  489. } catch {}
  490. }
  491. } else {
  492. //lazy - need to figure out way to handle on dom iter
  493. location.reload(true);
  494. // for (let i = 0; i < clickableImages.length; i++) {
  495. // try {
  496. // clickableImages[i].click();
  497. // clickableImages[i].click();
  498. // } catch {}
  499. // }
  500. }
  501. }
  502.  
  503.  
  504. function linksInNewTab() {
  505. const links = document.getElementsByTagName("a");
  506. for (let i = 0; i < links.length; i++) {
  507. links[i].setAttribute("target", "_blank");
  508. links[i].setAttribute("rel", "noreferrer");
  509. }
  510. }
  511. function blockContent(filters) {
  512. const blockFilters = filters;
  513. const posts = document.getElementsByClassName("post-listing");
  514. const comments = document.getElementsByClassName("comment");
  515. var blockedCount = 0;
  516. for (let y = 0; y < blockFilters.length; y++) {
  517. if (blockFilters[y].length >= 1)
  518. {
  519. blockFilters[y] = addslashes(blockFilters[y]);
  520. for (let i = 0; i < posts.length; i++) {
  521. if (posts[i].innerHTML.toLowerCase().indexOf(blockFilters[y].toLowerCase()) !== -1) {
  522. blockedCount++
  523. posts[i].setAttribute("style", "display: none !important;");
  524. //posts[i].innerHTML = "<div class='card small'>Post blocked due to filter: " + blockFilters[y] + "</div>"
  525. }
  526. }
  527. for (let x = 0; x < comments.length; x++) {
  528. if (comments[x].innerHTML.toLowerCase().indexOf(blockFilters[y].toLowerCase()) !== -1) {
  529. blockedCount++
  530. comments[x].setAttribute("style", "display: none !important;");
  531. //comments[x].innerHTML = "<div class='card small'>Post blocked due to filter: " + blockFilters[y] + "</div>"
  532. }
  533. }
  534. }
  535. }
  536. localStorage.setItem("currentBlockCount", blockedCount);
  537. ltLog("content blocking has blocked: " + blockedCount + " posts.", 2)
  538. }
  539. function addslashes( str ) {
  540. return (str + '').replace(/[\\"']/g, '\\$&').replace(/\u0000/g, '\\0');
  541. }
  542.  
  543. // LemmyTools
  544.  
  545. //check if first run or load saved settings
  546. let settings = options(2);
  547.  
  548.  
  549. function checkedIfTrue(val) {
  550. return val ? "checked" : "";
  551. }
  552.  
  553. /* Script */
  554. let url = document.location.href;
  555. window.onload = () => {
  556. if (settings.showAllImages) allImages(true); // initial expansion
  557. const body = document.querySelector("body");
  558. const observer = new MutationObserver((_) => {
  559. if (url !== document.location.href) {
  560. url = document.location.href;
  561. if (settings.showAllImages) {
  562. // todo there has to be a better way to wait for the content to be loaded …
  563. var imagesTimer = setTimeout(allImages, 5000);
  564. clearTimeout(imagesTimer);
  565. //allImages(true);
  566. }
  567. }
  568. });
  569. observer.observe(body, { childList: true, subtree: true });
  570. };
  571. let count = 0;
  572. let eIcheck = checkedIfTrue(settings.expandImages);
  573. let hSBcheck = checkedIfTrue(settings.hideSideBar);
  574. let hoverCheck = checkedIfTrue(settings.hoverCheck);
  575. let unblurCheck = checkedIfTrue(settings.unblurNSFW);
  576. let blockContentCheck = checkedIfTrue(settings.blockContent);
  577. let widthPixelCheck = checkedIfTrue(settings.widthPixels);
  578. let widthPercentCheck = checkedIfTrue(!settings.widthPixels);
  579. let showAllImagesCheck = checkedIfTrue(settings.showAllImages);
  580. let linksCheck = checkedIfTrue(settings.linksInNewTab);
  581. let hideShowAllImagesButtonCheck = checkedIfTrue(
  582. settings.hideShowAllImagesButton
  583. );
  584.  
  585. //Option Divs
  586. //Is HomeInstance Manually Set For WorkAround
  587.  
  588. var hIAlertString = isHomeInstanceSet(homeInstance)
  589. ? ""
  590. : "<b style='color: red;'>Your Home Instance has not been manually set in the UserScript.</b><br />";
  591.  
  592. //Create Lemmy Tools Elements ----------------------------------------
  593.  
  594. const odiv = document.createElement("div");
  595. odiv.setAttribute("id", "ltOptions");
  596. odiv.setAttribute("style", "display: none;")
  597. odiv.classList.add("card", "border-secondary", "ltoptions");
  598. odiv.innerHTML = `
  599. <h4>LemmyTools ${ltVer} Options - <button class='pointer btn btn-secondary text-bg-primary' id='LTsaveoptions'>Save&Close</button></h4>
  600. </hr>
  601. <div class='table-responsive'>
  602. <table class='table'>
  603. <thead class='pointer'>
  604. <tr>
  605. <th>Option:</th>
  606. <th>Value:</th>
  607. </thead>
  608. </tr>
  609. <tbody>
  610. <tr>
  611. <td><b>LemmyTools Settings: </b></td>
  612. <td></td>
  613. </tr>
  614. <tr>
  615. <td><b>HomeInstance URL</b><br /> Make sure to edit the homeInstance variable of<br /> the UserScript for the
  616. remote instance Home button fix. (Temporary workaround).<br />(Ex:
  617. https://yourinstance.lemmy)<br />${hIAlertString}</td>
  618. <td><textarea id='option_homeInstance'>${settings.instance}</textarea></td>
  619. </tr>
  620. <tr>
  621. <td><b>LemmyTools bar window side</b><br /> - default: right</td>
  622. <td><select id="option_commposSide">
  623. <option value='${settings.commposSide}'>${settings.commposSide}</option>
  624. <option value='top'>top</option>
  625. <option value='right'>right</option>
  626. <option value='left'>left</option>
  627. </select></td>
  628. </tr>
  629. <tr>
  630. <td><b>LemmyTools bar vertical position </b><br />% from top [0-85] - default: 0</td>
  631. <td><textarea id='option_commposVertical'>${settings.commposVertical}</textarea></td>
  632. </tr>
  633. <tr>
  634. <td><b>Keep LemmyTools Bar Open</b><br />Works best for widescreen desktops.</td>
  635. <td><input type='checkbox' id='option_hoverCheck' ${hoverCheck} /></td>
  636. </tr>
  637. <tr>
  638. <td><br /><br /></td>
  639. <td></td>
  640. </tr>
  641. <tr>
  642. <td><b>Site Style and Behaviors:</b></td>
  643. <td></td>
  644. </tr>
  645. <tr>
  646. <td><b>Hide Lemmy Sidebars</b><br /> (Trending, ServerInfo, Communities)<br />
  647. </td>
  648. <td><input type='checkbox' id='option_hideSideBar' ${hSBcheck} /></td>
  649. </tr>
  650. <tr>
  651. <td><b>Expandable Images</b><br />Acts as an auto-expander and adds the ability to manually<br /> expand
  652. images by clicking and dragging.<br />Doubleclick to open full image.</td>
  653. <td><input type='checkbox' id='option_expandImages' ${eIcheck} /></td>
  654. </tr>
  655. <tr>
  656. <td><b>Auto Expand Size</b><br />Size of post image after opening a image post.<br /> Desktop Default: 50 /
  657. Mobile: 100</td>
  658. <td><textarea id='option_expandImagesize'>${settings.expandImagesize}</textarea>
  659. <br /> <label for="option_widthPixels">Pixels</label> <input type='radio' id='option_widthPixels' name="widthScaler" ${widthPixelCheck}/>
  660. <label for="option_widthPercent">Percent</label> <input type='radio' id='option_widthPercent' name="widthScaler" ${widthPercentCheck}/>
  661. </td>
  662. </tr>
  663. <tr>
  664. <td><b>Expand Image Speed</b><br />Speed multiplier for click&drag expanding images. If your images seem to
  665. expand<br /> too fast or slow, increase or decrease this value. [Values 0 to 1.0]<br /> Default: 0.50 </td>
  666. <td><textarea id='option_expandImageSpeed'>${settings.expandImageSpeed}</textarea></td>
  667. </tr>
  668. <tr>
  669. <td><b>Automatically open image posts</b><br /></td>
  670. <td><input type='checkbox' id='option_showAllImages'${showAllImagesCheck}/></td>
  671. </tr>
  672. <tr>
  673. <td><b>Hide the Show All Images button (when Auto open image posts is disabled)</b><br /></td>
  674. <td><input type='checkbox' id='option_hideShowAllImagesButton'${hideShowAllImagesButtonCheck}/></td>
  675. </tr>
  676. <tr>
  677. <td><b>Auto unblur NSFW images</b><br /></td>
  678. <td><input type='checkbox' id='option_unblurNSFW' ${unblurCheck} /></td>
  679. </tr>
  680. <tr>
  681. <td><b>Open ALL links in new tab.</b><br /></td>
  682. <td><input type='checkbox' id='option_linksInNewTab' ${linksCheck} /></td>
  683. </tr>
  684. <tr>
  685. <td><b>Content Blocking: (Blocks posts and comments matching your desired filters).</b><br /></td>
  686. <td><input type='checkbox' id='option_blockContent' ${blockContentCheck} /></td>
  687. </tr>
  688. <tr>
  689. <td>Add filters seperated by commas. Ex: filter,filter two,filter3 <br />
  690. <textarea style="width:100%; "id='option_blockFilters'>${settings.blockFilters}</textarea></td>
  691. <td>Posts/Comments blocked from filters on this page:<br />${localStorage.getItem("currentBlockCount")}</td>
  692. </tr>
  693. <tr>
  694. <td></td>
  695. <td></td>
  696. </tr>
  697. </tbody>
  698. </table>
  699. </div>
  700. <p> Tested on Lemmy Version: ${ltTestedVer} on Firefox. <br />
  701.  
  702. <hr /><Get it here: <a
  703. href='https://github.com/howdy-tsc/LemmyTools'>GitHub</a> or <a
  704. href='https://greasyfork.org/en/scripts/469169-lemmytools'>GreasyFork</a><br />Please submit issues to the GitHub
  705. for feature requests and problems: <a href='https://github.com/howdy-tsc/LemmyTools/issues'>GitHub LemmyTools
  706. Issues</a><br /></p>
  707.  
  708. <h5>Attributes/Credit: </h5>
  709. <li><b>@cwagner@lemmy.cwagner.me</b> - For coding, code cleanup, and mentoring.</li>
  710. <li><b>Charles Machalow - csm10495</b> - Coding contribution(s).</li>
  711. <li><b>jimmyhiggs337</b> - Coding contribution(s).</li>
  712. `;
  713.  
  714. //Adjust clickable area for mobile (add brandingString if desktop)
  715. let brandingString = "";
  716. if (mobile !== true) {
  717. brandingString =
  718. "<span id='brandingText' style='vertical-align: super !important; writing-mode: vertical-lr; text-orientation: mixed;'>LemmyTools</span>";
  719. }
  720.  
  721. //Comm divs
  722.  
  723.  
  724. const div = document.createElement("div");
  725. div.setAttribute("id", "myDiv");
  726. div.classList.add("ltcommsbar");
  727.  
  728. const touchdiv = document.createElement("div");
  729. touchdiv.setAttribute("id", "touchdiv");
  730.  
  731. const idiv = document.createElement("div");
  732. idiv.setAttribute("id", "searchdiv");
  733. idiv.classList.add("ltmenu", "border-secondary", "card");
  734. // todo on input
  735.  
  736. idiv.innerHTML = `
  737. <div id='ltActiveSearchDiv' class='ltActiveSearchDiv'>
  738. <header id='ltBarHeader' class='card-header'>
  739. <h6><a href=${settings.instance}>Home</a> - <a href='https://lemmyverse.net/communities' target='_new'>Find
  740. Subs</a> - <img id="lemmyLogo" width=22 height=22 class='targetImg'
  741. src='' />
  742. <img id="lemmyOptionsIcon" width=22 height=22 class='targetImg' alt"LemmyTools Options" title="LemmyTools Options"
  743. src="" />
  744.  
  745. </h6>
  746. </header><input type='text' id='commsearch' name='commsearchinput'
  747. placeholder='Search your subscriptions (or visted subs)' />
  748. <div id='ltBarSubHeader' class='clickAble'>LemmyTools ${ltVer}</div>
  749. <div style='clear:both;'></div>
  750. </div>
  751. <div id='ltPassiveSearchDiv' class='ltPassiveSearchDiv'><img width=30 height=30 class='targetImg'
  752. src='' />${brandingString}
  753. </div><br />`;
  754.  
  755.  
  756. let styleString = `
  757. .ltmenu {
  758. position: fixed;
  759. top: ${settings.commposVertical}%;
  760. ${settings.commposSide}: 0;
  761. font-size: .9rem;
  762. display: block;
  763. height: 100%;
  764. min-height: auto;
  765. z-index: 999;
  766. overflow: scroll;
  767. outline: 1px solid grey !important;
  768. }
  769.  
  770. .ltActiveSearchDiv {
  771. font-size: 0.9rem;
  772. width: 100%;
  773. }
  774.  
  775. .ltActiveSearchDiv img:hover{
  776. margin-top:-5px;
  777. border:10px yellow;
  778. }
  779.  
  780.  
  781. .ltmenu input {
  782. width: 100%;
  783. }
  784.  
  785. .ltPassiveSearchDiv {
  786. display: none;
  787. width: 100%;
  788. }
  789.  
  790. .post-listings .img-expanded {
  791. width: ${settings.expandImagesize}${settings.widthPixels ? "px" : "%"};
  792. max-height: none !important;
  793. }
  794.  
  795. #myDiv li {
  796. list-style-type: none;
  797. }
  798.  
  799. #myDiv hr {
  800. display: block;
  801. }
  802.  
  803. #searchdiv {
  804. ${settings.commposSide}: 0;
  805. position: fixed;
  806. height: 100%;
  807. min-height: auto;
  808. width: 240px;
  809. display: block;
  810. z-index: 999;
  811. overflow: scroll;
  812. transition-timing-function: ease;
  813. transition: ${settings.commposSide} .25s;
  814. transition-delay: 0, 0.25s;
  815. overflow: auto;
  816. -ms-overflow-style: none;
  817. scrollbar-width: none;
  818. }
  819.  
  820. .ltbutton {
  821. background-color: #ccffe5;
  822. }
  823.  
  824. .ltPassiveSearchDiv {
  825. display: none;
  826. }
  827.  
  828. .ltoptions {
  829. position: fixed;
  830. min-width: auto;
  831. min-height: auto;
  832. width: auto;
  833. height: 100%;
  834. top: 0;
  835. display: none;
  836. left: 0;
  837. overflow: scroll;
  838. z-index: 1000;
  839. padding: 0.5%;
  840. margin-top:35px;
  841. }
  842.  
  843. #myDiv::-webkit-scrollbar {
  844. display: none;
  845. }
  846.  
  847. #myDiv {
  848. -ms-overflow-style: none;
  849. scrollbar-width: none;
  850. }
  851.  
  852. #searchdiv::-webkit-scrollbar {
  853. display: none;
  854. }
  855. .userNote {
  856. margin-left: 0.5em;
  857. //padding-left: 0.25em;
  858. //padding-right: 0.25em;
  859. font-size: 0.9em;
  860. }
  861. .userTagger {
  862. border-radius: var(--bs-border-radius);
  863. }
  864. .userTagger::backdrop {
  865. background: rgba(0, 0, 0, 0.7);
  866. }
  867. form#userTag {
  868. display: flex;
  869. flex-direction: column;
  870. gap: 4px;
  871. }
  872. form#userTag label {
  873. width: 60px;
  874. }
  875. `;
  876.  
  877. if (settings.unblurNSFW) {
  878. styleString +=
  879. " .img-blur {filter: none !important; -webkit-filter: none !important; -moz-filter: none !important; -o-filter: none !important; -ms-filter: none !important;} .img-blur-icon {display: none !important;} ";
  880. } else {
  881. styleString +=
  882. " .img-blur {filter: blur !important; -webkit-filter: blur !important; -moz-filter: blur !important; -o-filter: blur !important; -ms-filter: blur !important;} .img-blur-icon {filter: blur !important; -webkit-filter: blur !important; -moz-filter: blur !important; -o-filter: blur !important; -ms-filter: blur !important;} ";
  883. }
  884.  
  885. if (settings.hideSideBar) {
  886. styleString +=
  887. ".container, .container-lg, .container-md, .container-sm, .container-xl { }" +
  888. ".col-md-8 {flex: 0 0 80% !important;max-width: 80%;}";
  889. } else {
  890. styleString +=
  891. ".container, .container-lg, .container-md, .container-sm, .container-xl {}";
  892. }
  893. if (!settings.hoverCheck) {
  894. styleString += `
  895. #myDiv:not(:hover) {
  896. animation: showNavOut 500ms ease-in-out both;
  897. display: none;
  898. height: 0;
  899. transition-timing-function: ease;
  900. transition: height;
  901. transition-duration: 1.0;
  902. transition-delay: 0.5s;
  903. }
  904.  
  905. .ltPassiveSearchDiv {
  906. display: block;
  907. float: ${settings.reverseSide};
  908. padding-${settings.commposSide}: 200px;
  909. }
  910.  
  911. #ltActiveSearchDiv {
  912. display: none;
  913. animation: showNav 500ms ease-in-out both;
  914. }
  915.  
  916. #sidebarSubscribed {
  917. display: none;
  918. }
  919.  
  920. #searchdiv {
  921. ${settings.commposSide}: -200px;
  922. position: fixed;
  923. height: 110px;
  924. min-height: auto;
  925. width: 240px;
  926. display: block;
  927. z-index: 999;
  928. overflow: auto;
  929. display: block;
  930. transition-timing-function: ease;
  931. transition: ${settings.commposSide},
  932. height;
  933. transition-duration: 0.25s, 0.25s;
  934. transition-delay: 0.25s, 0.25s;
  935. /*animation: showNavOut 250ms ease-in-out both;*/ /*Breaks things?*/
  936. }
  937.  
  938. #searchdiv:hover .ltActiveSearchDiv {
  939. display: block;
  940. }
  941.  
  942. #searchdiv:hover .ltPassiveSearchDiv {
  943. display: none;
  944. }
  945.  
  946. #searchdiv:hover {
  947. ${settings.commposSide}: 0;
  948. position: fixed;
  949. height: 100%;
  950. min-height: auto;
  951. width: 240px;
  952. display: block;
  953. z-index: 999;
  954. display: block;
  955. overflow: auto;
  956. }
  957.  
  958. #searchdiv:hover>#myDiv {
  959. ${settings.commposSide}: 0;
  960. word-wrap: break-word;
  961. overflow: auto;
  962. display: block;
  963. height: 100%;
  964. width: 240px;
  965. animation: showNav 500ms ease-in-out both;
  966. }
  967.  
  968. @keyframes showNav {
  969. from {
  970. opacity: 0;
  971. }
  972.  
  973. to {
  974. opacity: 1;
  975. }
  976. }
  977. @keyframes showNavOut { from {opacity: 1;} to {opacity: 0;}}`;
  978. } else {
  979. styleString +=
  980. " myDiv {visibility: visible; height: auto; width: auto; overflow:scroll !important;}";
  981. }
  982.  
  983. //For mobile layouts make the ltbar tab smaller
  984. styleString += `
  985. @media (max-width: 1199.98px) {
  986. #brandingText {
  987. display:none;
  988. }
  989. #searchdiv {
  990. height: 35px;
  991. }
  992. }`;
  993.  
  994.  
  995.  
  996.  
  997. if (settings.commposSide != "top")
  998. {
  999. div.innerHTML = "<div class='commsAreaStatic'></div>";
  1000. document.body.appendChild(idiv);
  1001. idiv.appendChild(div);
  1002. ltLog("here");
  1003. }
  1004. else if (settings.commposSide == "top")
  1005. {
  1006.  
  1007. const topDiv = document.createElement("div");
  1008. topDiv.setAttribute("id", "topDiv");
  1009. topDiv.innerHTML = `
  1010. <div class='topDivRoot card border-secondary'>
  1011.  
  1012. <div class='topDivSearch'>
  1013. <img id="lemmyLogo" width=22 height=22 class='targetImg' alt="${homeInstance}" title="${homeInstance}"
  1014. src='' />
  1015. <img id="lemmyOptionsIcon" width=22 height=22 class='targetImg' alt"LemmyTools Options" title="LemmyTools Options"
  1016. src="" />
  1017. <input type='text' id='commsearch' name='commsearchinput' placeholder='Search your subscriptions (or visted subs)' />
  1018. <a id='dropDownComms' href="#" alt="Show All Your Communities and Manage" title="Show All Your Communities and Manage" > (Show All) </a>
  1019. </div>
  1020. <div class='topDivBar'>
  1021. <div class='commsAreaStatic'>
  1022. </div>
  1023. </div>
  1024. </div>
  1025. `;
  1026. const topDivCommsBox = document.createElement("div");
  1027. topDivCommsBox.setAttribute("id", "topDivCommsBox");
  1028. topDivCommsBox.style.display = "none";
  1029. topDivCommsBox.innerHTML = `
  1030. <header id='ltBarHeader' class='card-header'>
  1031. <h5><a href=${settings.instance}>Home</a> - <a href='https://lemmyverse.net/communities' target='_new'>Find
  1032. Communities</a>
  1033. </h5>
  1034. </header>
  1035. LemmyTools ${ltVer}
  1036. <div class="commsAreaSearch">
  1037. </div>
  1038. `;
  1039. document.body.prepend(topDivCommsBox);
  1040. document.body.prepend(topDiv);
  1041.  
  1042. document.getElementById("dropDownComms").addEventListener("click", (e) => {
  1043. e.preventDefault();
  1044. dropDownComms = e;
  1045. window.scrollTo(0, 0);
  1046. dropDownComms.target.innerHTML =
  1047. dropDownComms.target.innerHTML == " (Show All) " ? dropDownComms.target.innerHTML = " (Hide All) " : dropDownComms.target.innerHTML = " (Show All) ";
  1048. topDivCommsBox.style.display =
  1049. topDivCommsBox.style.display == "none" ? topDivCommsBox.style.display = "block" : topDivCommsBox.style.display = "none";
  1050. searchComms(searchInput.value, communityArray);
  1051. });
  1052. styleString += `
  1053. #topDiv
  1054. {
  1055. z-index: 99999999 !important;
  1056. }
  1057. #topDiv input
  1058. {
  1059.  
  1060. }
  1061. .topDivSearch
  1062. {
  1063. min-width: 330px;
  1064. overflow:hidden;
  1065. -ms-overflow-style: none;
  1066. scrollbar-width: none;
  1067. z-index: 99999999 !important;
  1068. "mailx" stuck linux
  1069. }
  1070. #topDiv img:hover{
  1071. margin-top:-5px;
  1072. border:10px yellow;
  1073. }
  1074.  
  1075. .topDivBar
  1076. {
  1077. width: 100%;
  1078. overflow:hidden;
  1079. -ms-overflow-style: none;
  1080. scrollbar-width: none;
  1081. }
  1082. .topDivBar br {
  1083. display: none;
  1084. }
  1085. .topDivBar li, a {
  1086. display:inline-block;
  1087. padding-right: 5px;
  1088. margin-top: 4px;
  1089. }
  1090. .topDivBar hr {
  1091. display:none;
  1092. }
  1093. .topDivBar img {
  1094. display:none;
  1095. }
  1096. .topDivRoot
  1097. {
  1098. height:35px;
  1099. position: fixed;
  1100. top: 0;
  1101. left: 0;
  1102. width:100%;
  1103. overflow: hidden;
  1104. white-space: nowrap;
  1105. z-index: 99999999 !important;
  1106. border:2px;
  1107. display: inline-flex;
  1108. flex-direction: row;
  1109. }
  1110.  
  1111. #topDivCommsBox
  1112. {
  1113. float:left;
  1114. width: 100%;
  1115. height: 100%;
  1116. top: 35px;
  1117. left: 0;
  1118. }
  1119.  
  1120. #root
  1121. {
  1122. margin-top: 35px !important;
  1123. }
  1124.  
  1125. `;
  1126.  
  1127. }
  1128. // ADD MAIN CSS
  1129. document.head.appendChild(document.createElement("style")).innerHTML = styleString;
  1130. //add lemmytools elements to page
  1131. document.body.appendChild(odiv); //options
  1132. let commsAreaSearch = document.getElementsByClassName("commsAreaSearch");
  1133. let commsAreaStatic = document.getElementsByClassName("commsAreaStatic");
  1134. let dropDownComms = document.getElementById("dropDownComms");
  1135.  
  1136. const searchInput = document.getElementById("commsearch");
  1137. searchInput.addEventListener("input", (e) => {
  1138. e.preventDefault();
  1139. searchComms(searchInput.value, communityArray);
  1140. });
  1141. document.getElementById("lemmyOptionsIcon").addEventListener("click", (e) => {
  1142. e.preventDefault();
  1143. options(1);
  1144. });
  1145. document.getElementById("LTsaveoptions").addEventListener("click", (e) => {
  1146. e.preventDefault();
  1147. options(3);
  1148. });
  1149. document.getElementById("lemmyLogo").addEventListener("click", (e) => {
  1150. e.preventDefault();
  1151. window.location = homeInstance;
  1152. });
  1153.  
  1154. // document.onreadystatechange = () => {
  1155. // if (document.readyState === "interactive") {
  1156. // // document ready
  1157. // if (
  1158. // showImagesButton.value == "Hide All Images" ||
  1159. // settings.showAllImages
  1160. // ) {
  1161. // allImages(true);
  1162. // }
  1163. // }
  1164. // };
  1165.  
  1166. //Easier Subscribe Buttons ---------------------------
  1167. ltLog("url is " + url);
  1168. let rCommunityArray = [];
  1169. //Browsing remote instance
  1170. setInterval(function () {
  1171. const broken = url.split("/c/");
  1172. const site = broken[0].replace("https://", "");
  1173. let community = broken[1];
  1174. let communityName = community;
  1175. try {
  1176. const broken2 = community.split("?");
  1177. community = broken2[0];
  1178. communityName =
  1179. community.indexOf("@") > -1 ? community : community + "@" + site;
  1180. } catch {}
  1181.  
  1182. const subString =
  1183. `${settings.instance}/search?q=!${communityName}&type=All&listingType=All&page=1`.replace(
  1184. "#",
  1185. ""
  1186. );
  1187.  
  1188. if (notHomeAndInCommunity(url)) {
  1189. ltLog(`On remote instance community - DIRECT - Button to: ${subString}`);
  1190. rCommunityArray = update(community, communityName, subString);
  1191. rCommunityArray = [...new Set(rCommunityArray)];
  1192. rCommunityArray = rCommunityArray.reverse();
  1193. commsAreaStatic[0].innerHTML = rCommunityArray;
  1194. communityArray = rCommunityArray;
  1195. }
  1196. }, 1000);
  1197. tagUsers();
  1198. // Update homeInstance Comms for bar to use
  1199.  
  1200. let communityArray = [];
  1201. if (url.includes(settings.instance)) {
  1202. ltLog("home instance do bar");
  1203. document
  1204. .querySelectorAll('[class="list-inline-item d-inline-block"]')
  1205. .forEach(function (el) {
  1206. communityArray.push("<li>" + el.innerHTML.toLowerCase() + "</li>");
  1207. });
  1208. communityArray = [...new Set(communityArray)];
  1209. if (count === 0 || count == null) {
  1210. count = communityArray.length;
  1211. }
  1212.  
  1213. communityArray = communityArray.join("");
  1214.  
  1215. commsAreaStatic[0].innerHTML += communityArray;
  1216. if ( commsAreaStatic[0].innerHTML.length >= 20) {
  1217. ltLog("Got Results >= 20", LogDebug);
  1218. ltLog("setting localcomms localstore", LogDebug);
  1219. localStorage.setItem("localComms", communityArray);
  1220. localStorage.setItem("commsCount", count.toString()); // todo why store the count? communityArray.length everywhere should be easier
  1221. //force update the page
  1222. searchComms(searchInput.value, communityArray);
  1223. } else {
  1224. ltLog("get localcomms from localstore", LogDebug);
  1225. communityArray = localStorage.getItem("localComms");
  1226.  
  1227. commsAreaStatic[0].innerHTML += communityArray;
  1228. //If previous search display previous results
  1229. try {
  1230. let latestQueryString = localStorage.getItem("prevSearchCommsQueries");
  1231. let latestQueryArray = [];
  1232. latestQueryArray = latestQueryString.split(",");
  1233. if (currentUrl.indexOf(latestQueryArray[latestQueryArray.length - 1]) !== -1) {
  1234. searchComms(latestQueryArray[latestQueryArray.length - 1], communityArray);
  1235. }
  1236. else {
  1237. searchComms(searchInput.value, communityArray);
  1238. }
  1239. }
  1240. catch {
  1241. searchComms(searchInput.value, communityArray);
  1242. }
  1243. }
  1244. }
  1245. else {
  1246. ltLog("On Remote Instance - Bar", LogDebug);
  1247. }
  1248.  
  1249. //Expand Images----------------------------------------------
  1250.  
  1251. setInterval(function () {
  1252. if (settings.expandImages === true) {
  1253. let theImages = document.getElementsByClassName("img-expanded");
  1254. for (let i = 0; i < theImages.length; i++) {
  1255. theImages[i].addEventListener("mousedown", startDrag);
  1256. }
  1257.  
  1258. let posX;
  1259. let node;
  1260.  
  1261. function startDrag(e) {
  1262. e.preventDefault();
  1263.  
  1264. node = e.currentTarget;
  1265. node.style.cursor = "nwse-resize";
  1266. try {
  1267. node.closest("a").removeAttribute("href");
  1268. node.target.closest("a").setAttribute("overflow", "auto;");
  1269. node.preventDefault();
  1270. } catch {}
  1271.  
  1272. posX = e.clientX;
  1273. document.addEventListener("mousemove", resize);
  1274. document.addEventListener("mouseup", stopDrag);
  1275. }
  1276.  
  1277. function resize(e) {
  1278. e.preventDefault();
  1279. const nextPosX = e.pageX;
  1280. node.style.width =
  1281. node.offsetWidth +
  1282. (nextPosX - posX) * settings.expandImageSpeed +
  1283. "px";
  1284. posX = nextPosX;
  1285. }
  1286.  
  1287. function stopDrag(e) {
  1288. e.preventDefault();
  1289. node.style.cursor = "default";
  1290. document.removeEventListener("mousemove", resize);
  1291. document.removeEventListener("mouseup", stopDrag);
  1292. }
  1293. } // if expand images
  1294.  
  1295. //Removes the offset from images.
  1296. try {
  1297. if (settings.expandImages) {
  1298. removeClassByWildcard("offset-*");
  1299. }
  1300. } catch {}
  1301.  
  1302. //sidebar settings do
  1303. if (settings.hideSideBar === true) {
  1304. try {
  1305. const sidebarSubscribed = document.getElementById("sidebarContainer");
  1306. sidebarSubscribed.style.display = "none";
  1307. removeClassByWildcard("site-sideba*");
  1308.  
  1309. const serverInfo = document.getElementById("sidebarInfo");
  1310. serverInfo.style.display = "none";
  1311. } catch {}
  1312. }
  1313.  
  1314. //Show All Images Functionality on button toggle.
  1315. try {
  1316. let addImageButtonArea = document.querySelector(".post-listings");
  1317. let showImage = document.createElement("div");
  1318. showImage.innerHTML =
  1319. "<div class='col-auto'><input type='button' id='showAllImages' class='pointer btn btn-secondary text-bg-primary' value='Show All Images' /> </div>";
  1320. if (
  1321. addImageButtonArea.innerHTML.indexOf("showAllImages") === -1 &&
  1322. !settings.showAllImages &&
  1323. !settings.hideShowAllImagesButton
  1324. ) {
  1325. addImageButtonArea.prepend(showImage);
  1326. const showImagesButton = document.getElementById("showAllImages");
  1327. showImagesButton.addEventListener("click", function () {
  1328. if (showImagesButton.value === "Show All Images") {
  1329. showImagesButton.value = "Hide All Images";
  1330. allImages(true);
  1331. } else {
  1332. showImagesButton.value = "Show All Images";
  1333. allImages(false);
  1334. }
  1335. });
  1336. }
  1337. } catch {}
  1338. //Links Open In New Tab
  1339. if (settings.linksInNewTab == true)
  1340. {
  1341. linksInNewTab();
  1342. }
  1343. //Block Content
  1344. if (settings.blockContent == true)
  1345. {
  1346. blockContent(settings.blockFilters);
  1347. };
  1348. }, 500);
  1349. }
  1350. })();
  1351. function isLemmySite() {
  1352. const meta = document.querySelector('meta[name="Description"]');
  1353. return (
  1354. meta && meta.content === "Lemmy"
  1355. );
  1356. }