Auto Scroll Button

Adds a button to scroll down the page when new content is loaded or by a certain interval

Pada tanggal 07 Agustus 2022. Lihat %(latest_version_link).

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name         Auto Scroll Button
// @namespace    https://github.com/Jkker/tampermonkey-auto-scroll
// @version      1.1.0
// @description  Adds a button to scroll down the page when new content is loaded or by a certain interval
// @author       Jkker
// @license      MIT
// @match        *://*/*
// @icon         https://raw.githubusercontent.com/Jkker/tampermonkey-auto-scroll/master/src/icons/Unfold.svg
// @grant        none
// @supportURL   https://github.com/Jkker/tampermonkey-auto-scroll/issues
// ==/UserScript==

(function(global, factory) {
  typeof exports === "object" && typeof module !== "undefined" ? module.exports = factory() : typeof define === "function" && define.amd ? define(factory) : (global = typeof globalThis !== "undefined" ? globalThis : global || self, global["auto-scroll"] = factory());
})(this, function() {
  "use strict";
  const app = "";
  function noop() {
  }
  function run(fn) {
    return fn();
  }
  function blank_object() {
    return /* @__PURE__ */ Object.create(null);
  }
  function run_all(fns) {
    fns.forEach(run);
  }
  function is_function(thing) {
    return typeof thing === "function";
  }
  function safe_not_equal(a, b) {
    return a != a ? b == b : a !== b || (a && typeof a === "object" || typeof a === "function");
  }
  let src_url_equal_anchor;
  function src_url_equal(element_src, url) {
    if (!src_url_equal_anchor) {
      src_url_equal_anchor = document.createElement("a");
    }
    src_url_equal_anchor.href = url;
    return element_src === src_url_equal_anchor.href;
  }
  function is_empty(obj) {
    return Object.keys(obj).length === 0;
  }
  function append(target, node) {
    target.appendChild(node);
  }
  function insert(target, node, anchor) {
    target.insertBefore(node, anchor || null);
  }
  function detach(node) {
    node.parentNode.removeChild(node);
  }
  function element(name) {
    return document.createElement(name);
  }
  function text(data) {
    return document.createTextNode(data);
  }
  function space() {
    return text(" ");
  }
  function listen(node, event, handler, options) {
    node.addEventListener(event, handler, options);
    return () => node.removeEventListener(event, handler, options);
  }
  function attr(node, attribute, value) {
    if (value == null)
      node.removeAttribute(attribute);
    else if (node.getAttribute(attribute) !== value)
      node.setAttribute(attribute, value);
  }
  function children(element2) {
    return Array.from(element2.childNodes);
  }
  function set_input_value(input, value) {
    input.value = value == null ? "" : value;
  }
  let current_component;
  function set_current_component(component) {
    current_component = component;
  }
  const dirty_components = [];
  const binding_callbacks = [];
  const render_callbacks = [];
  const flush_callbacks = [];
  const resolved_promise = Promise.resolve();
  let update_scheduled = false;
  function schedule_update() {
    if (!update_scheduled) {
      update_scheduled = true;
      resolved_promise.then(flush);
    }
  }
  function add_render_callback(fn) {
    render_callbacks.push(fn);
  }
  const seen_callbacks = /* @__PURE__ */ new Set();
  let flushidx = 0;
  function flush() {
    const saved_component = current_component;
    do {
      while (flushidx < dirty_components.length) {
        const component = dirty_components[flushidx];
        flushidx++;
        set_current_component(component);
        update(component.$$);
      }
      set_current_component(null);
      dirty_components.length = 0;
      flushidx = 0;
      while (binding_callbacks.length)
        binding_callbacks.pop()();
      for (let i = 0; i < render_callbacks.length; i += 1) {
        const callback = render_callbacks[i];
        if (!seen_callbacks.has(callback)) {
          seen_callbacks.add(callback);
          callback();
        }
      }
      render_callbacks.length = 0;
    } while (dirty_components.length);
    while (flush_callbacks.length) {
      flush_callbacks.pop()();
    }
    update_scheduled = false;
    seen_callbacks.clear();
    set_current_component(saved_component);
  }
  function update($$) {
    if ($$.fragment !== null) {
      $$.update();
      run_all($$.before_update);
      const dirty = $$.dirty;
      $$.dirty = [-1];
      $$.fragment && $$.fragment.p($$.ctx, dirty);
      $$.after_update.forEach(add_render_callback);
    }
  }
  const outroing = /* @__PURE__ */ new Set();
  function transition_in(block, local) {
    if (block && block.i) {
      outroing.delete(block);
      block.i(local);
    }
  }
  function mount_component(component, target, anchor, customElement) {
    const { fragment, on_mount, on_destroy, after_update } = component.$$;
    fragment && fragment.m(target, anchor);
    if (!customElement) {
      add_render_callback(() => {
        const new_on_destroy = on_mount.map(run).filter(is_function);
        if (on_destroy) {
          on_destroy.push(...new_on_destroy);
        } else {
          run_all(new_on_destroy);
        }
        component.$$.on_mount = [];
      });
    }
    after_update.forEach(add_render_callback);
  }
  function destroy_component(component, detaching) {
    const $$ = component.$$;
    if ($$.fragment !== null) {
      run_all($$.on_destroy);
      $$.fragment && $$.fragment.d(detaching);
      $$.on_destroy = $$.fragment = null;
      $$.ctx = [];
    }
  }
  function make_dirty(component, i) {
    if (component.$$.dirty[0] === -1) {
      dirty_components.push(component);
      schedule_update();
      component.$$.dirty.fill(0);
    }
    component.$$.dirty[i / 31 | 0] |= 1 << i % 31;
  }
  function init(component, options, instance2, create_fragment2, not_equal, props, append_styles, dirty = [-1]) {
    const parent_component = current_component;
    set_current_component(component);
    const $$ = component.$$ = {
      fragment: null,
      ctx: null,
      props,
      update: noop,
      not_equal,
      bound: blank_object(),
      on_mount: [],
      on_destroy: [],
      on_disconnect: [],
      before_update: [],
      after_update: [],
      context: new Map(options.context || (parent_component ? parent_component.$$.context : [])),
      callbacks: blank_object(),
      dirty,
      skip_bound: false,
      root: options.target || parent_component.$$.root
    };
    append_styles && append_styles($$.root);
    let ready = false;
    $$.ctx = instance2 ? instance2(component, options.props || {}, (i, ret, ...rest) => {
      const value = rest.length ? rest[0] : ret;
      if ($$.ctx && not_equal($$.ctx[i], $$.ctx[i] = value)) {
        if (!$$.skip_bound && $$.bound[i])
          $$.bound[i](value);
        if (ready)
          make_dirty(component, i);
      }
      return ret;
    }) : [];
    $$.update();
    ready = true;
    run_all($$.before_update);
    $$.fragment = create_fragment2 ? create_fragment2($$.ctx) : false;
    if (options.target) {
      if (options.hydrate) {
        const nodes = children(options.target);
        $$.fragment && $$.fragment.l(nodes);
        nodes.forEach(detach);
      } else {
        $$.fragment && $$.fragment.c();
      }
      if (options.intro)
        transition_in(component.$$.fragment);
      mount_component(component, options.target, options.anchor, options.customElement);
      flush();
    }
    set_current_component(parent_component);
  }
  class SvelteComponent {
    $destroy() {
      destroy_component(this, 1);
      this.$destroy = noop;
    }
    $on(type, callback) {
      const callbacks = this.$$.callbacks[type] || (this.$$.callbacks[type] = []);
      callbacks.push(callback);
      return () => {
        const index = callbacks.indexOf(callback);
        if (index !== -1)
          callbacks.splice(index, 1);
      };
    }
    $set($$props) {
      if (this.$$set && !is_empty($$props)) {
        this.$$.skip_bound = true;
        this.$$set($$props);
        this.$$.skip_bound = false;
      }
    }
  }
  const ArrowBackIcon = "";
  const SettingsIcon = "";
  const StopIcon = "";
  const AutoScrollIcon = "";
  const ClearIcon = "";
  const CheckIcon = "";
  const CloseIcon = "";
  function create_if_block_1(ctx) {
    let button0;
    let img;
    let img_src_value;
    let t0;
    let button1;
    let t1;
    let t2;
    let button2;
    let t3;
    let t4;
    let button3;
    let t5;
    let mounted;
    let dispose;
    return {
      c() {
        button0 = element("button");
        img = element("img");
        t0 = space();
        button1 = element("button");
        t1 = text("Auto");
        t2 = space();
        button2 = element("button");
        t3 = text("500ms");
        t4 = space();
        button3 = element("button");
        t5 = text("1.5s");
        if (!src_url_equal(img.src, img_src_value = ctx[1] ? CloseIcon : SettingsIcon))
          attr(img, "src", img_src_value);
        attr(img, "alt", "Settings");
        attr(button0, "style", ctx[12].button);
        attr(button1, "style", ctx[12].button);
        attr(button2, "style", ctx[12].button);
        attr(button3, "style", ctx[12].button);
      },
      m(target, anchor) {
        insert(target, button0, anchor);
        append(button0, img);
        insert(target, t0, anchor);
        insert(target, button1, anchor);
        append(button1, t1);
        insert(target, t2, anchor);
        insert(target, button2, anchor);
        append(button2, t3);
        insert(target, t4, anchor);
        insert(target, button3, anchor);
        append(button3, t5);
        if (!mounted) {
          dispose = [
            listen(button0, "click", ctx[6]),
            listen(button1, "click", ctx[11]("resize")),
            listen(button2, "click", ctx[11]("interval", 500)),
            listen(button3, "click", ctx[11]("interval", 1500))
          ];
          mounted = true;
        }
      },
      p(ctx2, dirty) {
        if (dirty & 2 && !src_url_equal(img.src, img_src_value = ctx2[1] ? CloseIcon : SettingsIcon)) {
          attr(img, "src", img_src_value);
        }
      },
      d(detaching) {
        if (detaching)
          detach(button0);
        if (detaching)
          detach(t0);
        if (detaching)
          detach(button1);
        if (detaching)
          detach(t2);
        if (detaching)
          detach(button2);
        if (detaching)
          detach(t4);
        if (detaching)
          detach(button3);
        mounted = false;
        run_all(dispose);
      }
    };
  }
  function create_if_block(ctx) {
    let div1;
    let label;
    let t0;
    let t1;
    let div0;
    let input;
    let t2;
    let button;
    let img;
    let img_src_value;
    let t3;
    let style;
    let mounted;
    let dispose;
    return {
      c() {
        div1 = element("div");
        label = element("label");
        t0 = text("Stop scrolling if");
        t1 = space();
        div0 = element("div");
        input = element("input");
        t2 = space();
        button = element("button");
        img = element("img");
        t3 = space();
        style = element("style");
        style.textContent = "#auto-scroll-wait-until-element-input::placeholder{color:rgba(255, 255, 255, 0.8);}";
        attr(label, "style", ctx[12].inputLabel);
        attr(label, "for", "auto-scroll-wait-until-element-input");
        attr(input, "id", "auto-scroll-wait-until-element-input");
        attr(input, "style", ctx[12].input);
        input.autofocus = true;
        attr(input, "placeholder", "text appears");
        if (!src_url_equal(img.src, img_src_value = ctx[3] ? ClearIcon : CheckIcon))
          attr(img, "src", img_src_value);
        attr(img, "alt", "Clear input");
        attr(button, "style", ctx[12].inputButton);
        attr(div0, "style", ctx[12].inputContainer);
        attr(div1, "style", ctx[12].menu);
      },
      m(target, anchor) {
        insert(target, div1, anchor);
        append(div1, label);
        append(label, t0);
        append(div1, t1);
        append(div1, div0);
        append(div0, input);
        set_input_value(input, ctx[2]);
        append(div0, t2);
        append(div0, button);
        append(button, img);
        append(div0, t3);
        append(div0, style);
        input.focus();
        if (!mounted) {
          dispose = [
            listen(input, "input", ctx[13]),
            listen(input, "change", ctx[8]),
            listen(input, "input", ctx[9]),
            listen(input, "keypress", ctx[14]),
            listen(button, "click", function() {
              if (is_function(ctx[3] ? ctx[7] : ctx[8]))
                (ctx[3] ? ctx[7] : ctx[8]).apply(this, arguments);
            })
          ];
          mounted = true;
        }
      },
      p(new_ctx, dirty) {
        ctx = new_ctx;
        if (dirty & 4 && input.value !== ctx[2]) {
          set_input_value(input, ctx[2]);
        }
        if (dirty & 8 && !src_url_equal(img.src, img_src_value = ctx[3] ? ClearIcon : CheckIcon)) {
          attr(img, "src", img_src_value);
        }
      },
      d(detaching) {
        if (detaching)
          detach(div1);
        mounted = false;
        run_all(dispose);
      }
    };
  }
  function create_fragment(ctx) {
    let div1;
    let div0;
    let t0;
    let button;
    let img;
    let img_src_value;
    let div0_style_value;
    let t1;
    let mounted;
    let dispose;
    let if_block0 = ctx[0] && create_if_block_1(ctx);
    let if_block1 = ctx[0] && ctx[1] && create_if_block(ctx);
    return {
      c() {
        div1 = element("div");
        div0 = element("div");
        if (if_block0)
          if_block0.c();
        t0 = space();
        button = element("button");
        img = element("img");
        t1 = space();
        if (if_block1)
          if_block1.c();
        if (!src_url_equal(img.src, img_src_value = ctx[4] ? StopIcon : ctx[0] ? ArrowBackIcon : AutoScrollIcon))
          attr(img, "src", img_src_value);
        attr(img, "alt", "Logo");
        attr(button, "style", ctx[12].button);
        attr(div0, "style", div0_style_value = ctx[12].container + `left: ${ctx[0] ? "0" : "-8px"};`);
      },
      m(target, anchor) {
        insert(target, div1, anchor);
        append(div1, div0);
        if (if_block0)
          if_block0.m(div0, null);
        append(div0, t0);
        append(div0, button);
        append(button, img);
        append(div1, t1);
        if (if_block1)
          if_block1.m(div1, null);
        if (!mounted) {
          dispose = listen(button, "click", function() {
            if (is_function(ctx[4] ? ctx[10] : ctx[5]))
              (ctx[4] ? ctx[10] : ctx[5]).apply(this, arguments);
          });
          mounted = true;
        }
      },
      p(new_ctx, [dirty]) {
        ctx = new_ctx;
        if (ctx[0]) {
          if (if_block0) {
            if_block0.p(ctx, dirty);
          } else {
            if_block0 = create_if_block_1(ctx);
            if_block0.c();
            if_block0.m(div0, t0);
          }
        } else if (if_block0) {
          if_block0.d(1);
          if_block0 = null;
        }
        if (dirty & 17 && !src_url_equal(img.src, img_src_value = ctx[4] ? StopIcon : ctx[0] ? ArrowBackIcon : AutoScrollIcon)) {
          attr(img, "src", img_src_value);
        }
        if (dirty & 1 && div0_style_value !== (div0_style_value = ctx[12].container + `left: ${ctx[0] ? "0" : "-8px"};`)) {
          attr(div0, "style", div0_style_value);
        }
        if (ctx[0] && ctx[1]) {
          if (if_block1) {
            if_block1.p(ctx, dirty);
          } else {
            if_block1 = create_if_block(ctx);
            if_block1.c();
            if_block1.m(div1, null);
          }
        } else if (if_block1) {
          if_block1.d(1);
          if_block1 = null;
        }
      },
      i: noop,
      o: noop,
      d(detaching) {
        if (detaching)
          detach(div1);
        if (if_block0)
          if_block0.d();
        if (if_block1)
          if_block1.d();
        mounted = false;
        dispose();
      }
    };
  }
  function instance($$self, $$props, $$invalidate) {
    const getLastElementByText = (text2) => {
      const iter = document.evaluate(`//*[contains(text(),"${text2}")]`, document, null, XPathResult.ORDERED_NODE_ITERATOR_TYPE, null);
      let curr, prev;
      try {
        curr = iter.iterateNext();
        while (curr) {
          prev = curr;
          curr = iter.iterateNext();
        }
      } catch (e) {
        console.error(`Error: Document tree modified during iteration ${e}`);
      }
      return prev;
    };
    let menuOpen = false;
    const toggleMenu = () => $$invalidate(0, menuOpen = !menuOpen);
    let settingsOpen = false;
    const toggleSettings = () => $$invalidate(1, settingsOpen = !settingsOpen);
    let inputValue = "";
    let untilTarget;
    const clearScrollUntilCondition = () => $$invalidate(3, untilTarget = void 0);
    const activateScrollUntilCondition = () => {
      $$invalidate(3, untilTarget = inputValue);
      toggleSettings();
    };
    const onInput = () => {
      $$invalidate(3, untilTarget = void 0);
    };
    let lastFound = null;
    function checkStopCondition() {
      if (!untilTarget)
        return false;
      const last = getLastElementByText(untilTarget);
      if (!last)
        return false;
      if (last !== lastFound) {
        lastFound = last;
        console.log("found", last);
        last.scrollIntoView({ behavior: "smooth" });
        return true;
      } else {
        return false;
      }
    }
    const scroll = () => {
      if (checkStopCondition()) {
        cancelCurrentScroll();
        return;
      }
      window.scroll({
        top: document.body.scrollHeight,
        left: 0,
        behavior: "smooth"
      });
    };
    const resizeObserver = new ResizeObserver(() => setTimeout(scroll, 100));
    let intervalId;
    let scrolling = false;
    const startScroll = (type, interval = 1e3) => {
      if (type === "resize")
        resizeObserver.observe(document.body);
      else if (type === "interval")
        intervalId = setInterval(scroll, interval);
      $$invalidate(4, scrolling = type);
      $$invalidate(0, menuOpen = !menuOpen);
    };
    const cancelScroll = (type) => {
      if (type === "resize")
        resizeObserver.unobserve(document.body);
      else if (type === "interval")
        clearInterval(intervalId);
      $$invalidate(4, scrolling = false);
    };
    const cancelCurrentScroll = () => {
      if (scrolling)
        cancelScroll(scrolling);
    };
    const toggleScroll = (type, intervalSecond = 1e3) => () => {
      if (scrolling) {
        cancelScroll(type);
      } else {
        startScroll(type, intervalSecond);
      }
    };
    const styles = {
      button: "height: 40px; padding: 8px; display: flex; justify-content: center; align-items: center; border: none; outline: none; background: transparent; cursor: pointer; color: white",
      menu: "position: fixed; bottom: 62px; left: 6px; background: rgba(0,0,0,0.4); backdrop-filter: blur(10px); padding: 8px; border-radius: 8px; padding-bottom: 12px;",
      inputLabel: "color: white; font-size: 15px",
      inputContainer: "display: flex; position: relative; margin-top: 8px",
      input: "height: 24px; border: 1px solid white; outline: none; background: transparent; border-radius: 4px; padding-left: 4px; padding-right: 32px; color: white",
      inputButton: "height: 100%; width: 28px; padding: 8px; display: flex; justify-content: center; align-items: center; border: none; outline: none; cursor: pointer; position: absolute; top: 0; right: 0; border-left: 1px solid white; background: transparent",
      container: "position: fixed; bottom: 16px;  z-index: 99999999; background: rgba(0,0,0,0.4); backdrop-filter: blur(10px); display: flex; border-radius: 0 20px 20px 0;"
    };
    function input_input_handler() {
      inputValue = this.value;
      $$invalidate(2, inputValue);
    }
    const keypress_handler = (e) => {
      if (e.key === "Enter")
        activateScrollUntilCondition();
    };
    return [
      menuOpen,
      settingsOpen,
      inputValue,
      untilTarget,
      scrolling,
      toggleMenu,
      toggleSettings,
      clearScrollUntilCondition,
      activateScrollUntilCondition,
      onInput,
      cancelCurrentScroll,
      toggleScroll,
      styles,
      input_input_handler,
      keypress_handler
    ];
  }
  class App extends SvelteComponent {
    constructor(options) {
      super();
      init(this, options, instance, create_fragment, safe_not_equal, {});
    }
  }
  const main = new App({
    target: document.body
  });
  return main;
});