Jigs' Blog

My name is John Johnson.

But my friends call me Jigs!

How to Quickly Create Your Own Feature Rich Bookmarklets

So, you want to allow users to run arbitrary code on a page you don’t have access to or influence over. One solution is to create a browser plugin, but there is another solution that is actually quite old, but still well supported and it’s called a bookmarklet. The way you generally use a bookmarklet as a user is to drag it to your bookmarks bar and click it on a given page.

So, if you’ve ever wondered how to make a bookmarklet it’s basically a link whose href looks like this href="javascript:<your-code-here>;" where <your-code-here> is a JavaScript expression. Here’s an example bookmarklet with its html that appends Hello World to the document.

<a
  href="javascript:document.body.appendChild(document.createTextNode('Hello World'));"
  >Say Hello</a
>

Creating a feature rich bookmarklet may seem hard due to it being a single expression that has to go inside the href’s javascript: ; statement, but there are a few tricks we can use to make it much simpler.

If you wanted to make a more complex bookmarklet you would have to wrap it in an Imediately Invoked Function Expression or IIFE which basically is just a function expression that invokes/calls itself like so:

(function() {
  /* Your code here */
})();

You can use this method directly by just writing a multiline function inside your a tag’s href attribute surrounded by the previously pointed out javascript: ; statement. Your editor probably won’t have syntax highlighting and linting features available inside that string, so development could be tedious. So, you can get around this by creating a function and programmatically turning it into a string and wrapping that with parenthesis to turn it into an IIFE and inserting that into the a’s href. That could like like this:

That’s pretty awesome! Now anything we write in the function main will be code that is run by our bookmarklet. We just have to distribute the a tag code given in the textarea. This is pretty much the finished product for productively creating bookmarklets, but there is a warning ahead when using this method and also some cool bookmarklets I have made in the past as examples down below.

Try clicking the above link check your browser console to see it run. Also, hover over the link in the preview and notice that all the javascript appears on one line. This property of a bookmarklet actually adds a few complications to writing your functionality in the main function. For example if we place a single line comment in our main function it will no longer work. Another way that things could go wrong is if your code does not have semicolons. Though semicolons are avoidable most the time in JavaScript semicolons are not avoidable in the case of the single line bookmarklet. Here’s an example of a comment in main that ruins the bookmarklet’s execution.

WARNING: This code example is broken on purpose for explanatory purposes:

So, this code doesn’t work. Hover over the link in the preview and notice how it comments the whole rest of the code. You could also check your browser console for errors when you click the link which in Chrome states “Uncaught SyntaxError: unexpected end of input”. However, though you can’t use single line comments you should be able to use multiline comments just fine.

So, that’s basically it. Go have some fun and create something cool! Just to give you some ideas here are some bookmarklets I have made in the past.

codepen

<body translate="no">
  <div>Drag the link to your bookmarks bar and click it.</div>
  <details>
    <summary>Click here for Usage Help</summary>
    <p>
      <b>Polling:</b> with polling turned on every 500 milliseconds it will
      search for new video elements on the page and will change their playback
      rates.
    </p>
    <p>
      <b>Skip:</b> could be used to skip ads. Only works if the video is
      playing. Hasn't been tested on many videos.
    </p>
  </details>
  <script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.2.1/jquery.min.js"></script>
  <script id="rendered-js">
    // edit main function and add all your javascript into it.
    var nextId = idGen();
    // you can edit this call to give your bookmarklet a different title and such
    toBookmarklet({
      mainFn: main,
      title: "Video Playback Rate Changer",
      id: "bookmarklet1",
      name: "Video Playback Rate Changer",
      description: `Creates UI for changing Video Playback rates.`,
    });

    async function main() {
      const mainId = "PlaybackChanger-v0-0-0";
      function log(...args) {
        window[mainId + "debugOn"] && console.log(...args);
      }
      const timeout = ms => {
        let id, reject;
        const p = new Promise((res, rej) => {
          reject = rej;
          id = setTimeout(res, ms);
        });
        p.clear = () => (clearTimeout(id), reject());
        return p;
      };
      function appendStyle(css) {
        var style = document.createElement("style");
        style.type = "text/css";
        style.innerHTML = css;
        document.getElementsByTagName("head")[0].appendChild(style);
      }
      function AsyncLock() {
        const p = () => new Promise(next => (nextIter = next));
        var nextIter,
          next = p();
        const nextP = () => {
          const result = next;
          next = result.then(() => p());
          return result;
        };
        nextIter();
        return Object.assign(
          {},
          {
            async *[Symbol.asyncIterator]() {
              try {
                yield nextP();
              } finally {
                nextIter();
              }
            },
          }
        );
      }
      const pollLock = AsyncLock();
      async function* poller(ms = 500) {
        for await (const _ of pollLock) {
          yield;
          while (true) {
            await timeout(ms);
            yield;
          }
        }
      }
      async function changePlaybackRate(rate) {
        window[mainId + "currentPoller"] &&
          window[mainId + "currentPoller"].return();
        const el = document.getElementById("PlaybackRate");
        el.innerHTML = "";
        el.appendChild(document.createTextNode("" + rate.toFixed(2)));
        window[mainId + "currentPoller"] = poller();
        for await (const _ of window[mainId + "currentPoller"]) {
          log("changing video players to playback rate of: " + rate.toFixed(2));
          const videos = document.querySelectorAll("video");
          videos.forEach(v => {
            if (v.playbackRate != rate) {
              v.playbackRate = rate;
            }
          });
          if (!window[mainId + "pollingOpen"]) {
            return;
          }
        }
      }
      function getPlaybackRate() {
        const videos = document.querySelectorAll("video");
        const playingVid = [...videos].find(v => !v.paused);
        return (
          window[mainId + "rate"] ||
          Number(
            playingVid
              ? playingVid.playbackRate
              : videos[0]
              ? videos[0].playbackRate
              : 1
          )
        );
      }
      const div = document.createElement("div");
      if (document.getElementById(mainId)) {
        document.getElementById(mainId).outerHTML = "";
      }
      document.body.appendChild(div);
      div.outerHTML = `
  <div id="${mainId}">
    <h1 id="PlaybackRate">${getPlaybackRate().toFixed(2)}</h1>
    <input id="PlaybackSetter" type="range" value="${getPlaybackRate()}"" list="tickmarks" min="0.25" max="3" step="0.25">
    <div class="PlaybackCheckboxes">
      <label for="PlaybackPoll">Polling:</label><br/>
      <input id="PlaybackPoll" type="checkbox" ${
        window[mainId + "pollingOpen"] ? "checked" : ""
      } />
    </div>
    <div class="PlaybackCheckboxes">
      <label for="PlaybackDebug">Debug:</label><br/>
      <input id="PlaybackDebug" type="checkbox" ${
        window[mainId + "debugOn"] ? "checked" : ""
      } />
    </div>
    <button id="${mainId}-skip">skip</button>
    <div id="${mainId}-close">X</div>
  </div>
  `;
      appendStyle(`
  #${mainId}-close {
    font-size: 20px;
    font-weight: 800;
    cursor: pointer;
  }

  #${mainId}-close:hover {
    color: #f0f0f0;
  }

  #${mainId} button {
    background-color: #f0f0f0;
    padding: 10px;
    color: black;
    border: 1px solid #ccc;
    border-radius: 3px;
  }

  #${mainId} button:hover {
    background-color: #fff;
    cursor: pointer;
  }

  #${mainId} {
    box-sizing: border-box;
    font-size: 10px;
    position: fixed;
    color: white;
    display: flex;
    flex-direction: row;
    justify-content: space-around;
    align-items: center;
    top: 0;
    right: 0;
    padding: 1em;
    opacity: 0.7;
    background-color: black;
    z-index: 99999;
    width: 500px;
    max-width: 100vw;
  }

  #${mainId} input[type=range] {
    -webkit-appearance: none; /* Hides the slider so that custom slider can be made */
    width: 100%; /* Specific width is required for Firefox. */
    background: transparent; /* Otherwise white in Chrome */
  }

  #${mainId} input[type=range]::-webkit-slider-thumb {
    -webkit-appearance: none;
  }

  #${mainId} input[type=range]:focus {
    outline: none; /* Removes the blue border. You should probably do some kind of focus styling for accessibility reasons though. */
  }

  #${mainId} input[type=range]::-ms-track {
    width: 100%;
    cursor: pointer;

    /* Hides the slider so custom styles can be added */
    background: transparent; 
    border-color: transparent;
    color: transparent;
  }

  #${mainId} input[type='range'] {
    -webkit-appearance: none !important;
    background-color: #555;
    height: 7px;
    width: 100%;
    padding: 4px;
    border-radius: 3.5px;
  }

  #${mainId} input[type='range']::-webkit-slider-thumb {
    -webkit-appearance: none !important;
    height: 14px;
    width: 14px;
    border: 1px solid #ccc;
    -webkit-border-radius: 50%;
    border-radius: 50%;
    background-color: #fcfcfc;
  }

  /* All the same stuff for Firefox */
  #${mainId} input[type=range]::-moz-range-thumb {
    height: 14px;
    width: 14px;
    border: 1px solid #ccc;
    -webkit-border-radius: 50%;
    border-radius: 50%;
    background-color: #fcfcfc;
  }

  /* All the same stuff for IE */
  #${mainId} input[type=range]::-ms-thumb {
    height: 14px;
    width: 14px;
    border: 1px solid #ccc;
    -webkit-border-radius: 50%;
    border-radius: 50%;
    background-color: #fcfcfc;
  }

  #${mainId} > * {
    margin: 0 0.5em;
    flex: auto;
  }

  #${mainId} * {
    box-sizing: border-box;
    font-family: -apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji" !important;
  }

  #PlaybackRate {
    font-size: 2.25em !important;
    font-weight: 800;
  }

  #PlaybackSetter {
    max-width: 100px;
    flex: 1 1 0;
    width: 100%;
    min-width: 50%; 
  }
  `);
      const el = document.getElementById("PlaybackSetter");
      let timer;
      el.addEventListener("input", async function(ev) {
        timer && timer.clear();
        const rate = Number(this.value);
        window[mainId + "rate"] = rate;
        changePlaybackRate(rate);
        try {
          timer = timeout(5000);
          await timer;
          document.getElementById(mainId).outerHTML = "";
        } catch (err) {}
      });
      document.getElementById("PlaybackPoll").addEventListener("input", ev => {
        window[mainId + "pollingOpen"] = ev.target.checked;
        if (!window[mainId + "pollingOpen"]) {
          window[mainId + "currentPoller"] &&
            window[mainId + "currentPoller"].return();
        } else {
          changePlaybackRate(window[mainId + "rate"]);
        }
      });
      document.getElementById("PlaybackDebug").addEventListener("input", ev => {
        window[mainId + "debugOn"] = ev.target.checked;
      });
      document
        .getElementById(`${mainId}-skip`)
        .addEventListener("click", () => {
          const v = [...document.querySelectorAll("video")].find(
            v => !v.paused
          );
          if (v) {
            v.currentTime = v.duration;
          }
        });
      document
        .getElementById(`${mainId}-close`)
        .addEventListener("click", () => {
          document.getElementById(mainId).outerHTML = "";
        });
    }

    // do not edit this function
    function toBookmarklet(options) {
      const defaults = {
        mainFn: main,
        title: "",
        id: nextId(),
        name: "",
        description: "",
      };
      options = $.extend({}, defaults, options);
      var { mainFn, title, id, name, description } = options;
      var html = $(`<div>
  <a id="link-${id}">${title}</a>
  <p>${description}</p>
  <textarea id="output-${id}"></textarea>
  <hr>
</div>`);
      var link = html.find(`#link-${id}`);
      link.attr(
        "href",
        `
  javascript:
  (${mainFn.toString()})()
  `
      );
      html.find(`#output-${id}`).val((link[0] || {}).outerHTML);
      html.appendTo(document.body);
    }

    function idGen(id = 0) {
      return () => id++;
    }
  </script>
</body>

codepen

<body translate="no">
  <p>
    Below is a bookmarklet that can be dragged to your bookmarks bar and clicked
    on any page to deploy nyan cats! Photo credit:
    <a href="https://valcreon.deviantart.com/art/Nyan-Cat-212131133"
      >Nyan cat credit</a
    >
  </p>
  Bookmarklet:
  <a
    href='javascript: (function () {
  var sheet = (function() {
	var style = document.createElement("style");
	style.appendChild(document.createTextNode(""));
	document.head.appendChild(style);

	return style.sheet;
})();

sheet.insertRule(`
.nyanCatWrapper {
  animation: linear 4s 1 slideInFromRight;
  width: 100%;
  position: fixed;
  z-index: 21345678;
  pointer-events: none;
}`);

sheet.insertRule(`.nyanCat {
  animation: linear 4s 1 opague;
  opacity: 0;
  background-image: url();
  width: 200px;
  height: 100px;
  background-size:     cover;
  background-repeat:   no-repeat;
  background-position: center center; 
}`);

sheet.insertRule(`@keyframes slideInFromRight {
  0% {
    transform: translate(100%);
    opacity: 1;
  }
  100% {
    transform: translate(-100%);
    opacity: 1;
  }
}`);

sheet.insertRule(`@keyframes opague {
  0% {
    opacity: 1;
  }
  100% {
    opacity: 1;
  }
}
`);

function getRandomIntInclusive(min, max) {
  min = Math.ceil(min);
  max = Math.floor(max);
  return Math.floor(Math.random() * (max - min + 1)) + min;
}

function spawnNyanCat () {
  var div = document.createElement("div");
  div.innerHTML = (`<div class="nyanCatWrapper" style="top: ${getRandomIntInclusive(0,90)}%"><div class="nyanCat"></div></div>`);
  document.body.appendChild(div);
  setTimeout(()=>div.parentNode.removeChild(div), 4000);
}

setInterval(spawnNyanCat, 500)

})()'
    >Deploy Nyan Cats</a
  >
</body>

codepen

<body translate="no">
  <h1>Better font bookmarklet</h1>
  <p>Add the bootstrap 4 font to the body</p>
  <p>
    Just drag this bookmarklet to your toolbar and click it to activate powers
  </p>
  <a
    href='javascript: (function () { document.querySelectorAll("*").forEach(function(el) { el.style.fontFamily = "-apple-system,system-ui,BlinkMacSystemFont,\"Segoe UI\",Roboto,\"Helvetica Neue\",Arial,sans-serif" }); })();'
    >Better font</a
  >

  <div id="rbm-extension-installed-rbwmlom"></div>
</body>

codepen

<body translate="no">
  <div>Drag the link to your bookmarks bar and click it.</div>
  <script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.2.1/jquery.min.js"></script>
  <script id="rendered-js">
    // edit main function and add all your javascript into it.
    var nextId = idGen();
    toBookmarklet({
      mainFn: main,
      title: "Narrow Body",
      id: "bookmarklet1",
      name: "Example Bookmarklet",
      description: `This is an example bookmarklet. Add Javascript to the main function to have it executed.`,
    });

    function main() {
      var style = document.createElement("style");
      style.type = "text/css";
      style.innerHTML = `
body {
  margin-left: auto;
  margin-right: auto;
  width: 600px;
}
  `;
      document.getElementsByTagName("head")[0].appendChild(style);
    }

    function toBookmarklet(options) {
      const defaults = {
        mainFn: main,
        title: "",
        id: nextId(),
        name: "",
        description: "",
      };
      options = $.extend({}, defaults, options);
      var { mainFn, title, id, name, description } = options;
      var html = $(`<div>
  <a id="link-${id}">${title}</a>
  <p>${description}</p>
  <textarea id="output-${id}"></textarea>
  <hr>
</div>`);
      var link = html.find(`#link-${id}`);
      link.attr(
        "href",
        `
  javascript:
  (${mainFn.toString()})()
  `
      );
      html.find(`#output-${id}`).val((link[0] || {}).outerHTML);
      html.appendTo(document.body);
    }

    function idGen(id = 0) {
      return () => id++;
    }
  </script>
</body>

codepen

<body translate="no">
  <div>Drag the link to your bookmarks bar and click it.</div>
  <script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.2.1/jquery.min.js"></script>
  <script id="rendered-js">
    // edit main function and add all your javascript into it.
    var nextId = idGen();
    toBookmarklet({
      mainFn: main,
      title: "Margin Top/Bottom",
      id: "bookmarklet1",
      name: "Example Bookmarklet",
      description: `This is an example bookmarklet. Add Javascript to the main function to have it executed.`,
    });

    function main() {
      var style = document.createElement("style");
      style.type = "text/css";
      style.innerHTML = `
body {
  margin-top: 20vh;
  margin-bottom: 20vh;
}
  `;
      document.getElementsByTagName("head")[0].appendChild(style);
    }

    function toBookmarklet(options) {
      const defaults = {
        mainFn: main,
        title: "",
        id: nextId(),
        name: "",
        description: "",
      };
      options = $.extend({}, defaults, options);
      var { mainFn, title, id, name, description } = options;
      var html = $(`<div>
  <a id="link-${id}">${title}</a>
  <p>${description}</p>
  <textarea id="output-${id}"></textarea>
  <hr>
</div>`);
      var link = html.find(`#link-${id}`);
      link.attr(
        "href",
        `
  javascript:
  (${mainFn.toString()})()
  `
      );
      html.find(`#output-${id}`).val((link[0] || {}).outerHTML);
      html.appendTo(document.body);
    }

    function idGen(id = 0) {
      return () => id++;
    }
  </script>
</body>