This is a collection of useful bookmarklets. If you find this helpful, consider starring it. If you know someone who may benefit from it, share it with them! If you want to add a bookmarklet or contribute in some way, please see CONTRIBUTING.md.
- Disclaimer
- What are bookmarklets?
- How to Use
- YouTube Standard Page/Embed Toggle
- YouTube Video Speed Adjuster
- YouTube Share Link Generator
- QR Code Generator
- Discord Messenger
- Discord Messenger v2 🔥
- Drag and Drop Website Editor
- Scroll to Bottom
- Continuously Scroll to Bottom
- Edit/Simplify Page Title
- Anonymous Instagram Viewer
I don't endorse any of these bookmarklets. Some of them may be against some Terms of Service, and I am not responsible for how you use these, and you are accountable for your actions. Please use them responsibly and at your own risk. No statements I make in any other place or time void this disclaimer.
Bookmarklets are pieces of Javascript code you can put in a bookmark in a web browser that supports them. When clicked, the Javascript code will run, executing whatever the code does. Be wary of bookmarks you get from strangers, and read through them carefully. Be especially careful about external links in them, as your browser cookies (which may contain sensitive information such as passwords) and other information could be sent to them. Don't worry, all bookmarklets on this page have been manually approved and are safe.
Tip: Most bookmarklets can be copied and pasted into your browser's console and run that way, too!
Note: Due to the security policy of some websites, some bookmarklets will not function on some websites.
- Create a bookmark in a browser that supports Javascript bookmarklets.
- Copy the full code into the URL section, and give it a name (I often use short names or letter abbreviations so I can fit more into my bookmark bar).
- Click the bookmarklet like you would with a regular bookmark. The code in the bookmarklet should execute.
Tip: If it doesn't work, check the browser console for errors.
#youtube #utility | v1.0.0 | ⬆ |
---|
If you are on a YouTube page such as https://www.youtube.com/watch?v=dQw4w9WgXcQ and click this bookmarklet, you will be taken to https://www.youtube.com/embed/dQw4w9WgXcQ. Inversely, if you are on https://www.youtube.com/embed/dQw4w9WgXcQ and click it, you will be taken to https://www.youtube.com/watch?v=dQw4w9WgXcQ. This is useful if I want to watch a video in a larger player but don't want to go full screen. It can also be used to bypass YouTube's "Disable your ad blocker" message, as it doesn't appear when viewing it from the embed URL.
javascript: (function() {
var url = window.location.href;
var youtubeURL = "https://www.youtube.com";
var embedURL = "/embed/";
var watchURL = "/watch?v=";
if (url.includes(watchURL)) {
var vidID = new URLSearchParams(window.location.search).get("v");
if (vidID) {
window.location.href = youtubeURL + "/embed/" + vidID;
} else {
alert("Error: Couldn't find the video's ID.");
}
} else if (url.includes(embedURL)) {
var videoID = url.split(embedURL)[1];
window.location.href = youtubeURL + "/watch?v=" + videoID;
} else if (!url.includes(youtubeURL)) {
alert("Error: The current URL is not a YouTube URL.");
} else {
alert("Error: The current URL is not a valid video.");
}
})();
#youtube #utility | v1.0.3 | ⬆ |
---|
Use this bookmarklet to set your videos to a faster speed or a specific (e.g. 2.8) speed.
Note: YouTube videos no won't play audio when the speed is higher than 4.
Tip: This can be used with many other video players, too! However, it will alter the first video on the page.
javascript:speed = prompt('Select Playback Rate:');
if (speed !== "") {
document.querySelector('video').playbackRate = speed;
}
window.stop();
If you have a favorite speed to watch videos at, use the below bookmarklet and change the "2" to your desired speed. Upon clicking, it will set the video to the specified speed, skipping the pop-up prompt.
javascript:document.querySelector('video').playbackRate = 2;
window.stop();
#youtube #utility | v1.0.0 | ⬆ |
---|
Copying YouTube short URLs (e.g. https://youtu.be/dQw4w9WgXcQ) has never been easy. You need to click on the Share button (sometimes nested within a 3 dots icon), let it load, and then click the copy button (and then click the X to exit out of the modal!). With this simple bookmarklet, you can simply click it and it'll copy the shortened youtu.be URL to your clipboard.
Tip: Change all or some of the "alert" actions to "console.log" if you want the process to be more seamless.
javascript: var url = window.location.href;
if (url.includes("youtube.com")) {
var vidID = new URLSearchParams(window.location.search).get("v");
if (vidID) {
navigator.clipboard.writeText("https://youtu.be/" + vidID);
alert(`Successfully copied video shortlink. (https://youtu.be/${vidID})`);
} else {
alert("Error: Video ID not found.");
}
} else {
alert("Error: This is not a YouTube site.");
}
#utility | v1.0.0 | ⬆ |
---|
This is a very useful bookmarklet for generating QR Codes of a website to then view on another device (with QR Code scanning abilities, of course). Credit goes to Jarón Berends and you can view it here on CodePen.
Help wanted! This bookmarklet seems complicated, and it can't run in the browser console (returns "SyntaxError: missing variable name"). If you can simplify it, that would be appreciated (see CONTRIBUTING.md).
javascript: (function() {
var % 20 qrSrc = 'https://chart.googleapis.com/chart%3Fchs=250x250%26cht=qr%26chl=' + encodeURIComponent(document.location.href), overlay = document.createElement('div'), os = overlay.style, img = document.createElement('img');
img.src = qrSrc;
os.position = 'fixed';
os.zIndex = 1000;
os.width = '100%25';
os.height = '100%25';
os.top = 0;
os.left = 0;
os.textAlign = 'center';
os.backgroundColor = 'rgba(0,0,0,0.9)';
img.style.marginTop = '100px';
overlay.appendChild(img);
document.body.appendChild(overlay);
overlay.addEventListener('click', function() {
document.body.removeChild(overlay);
})
})();
#discord #message #utility #performance | v1.2.0 | ⬆ |
---|
With this bookmarklet, you can send messages to people without booting up Discord (which is slow sometimes). There are two setup steps to get this working.
- Put your token into the area in the below code marked "PUT.YOUR_DISCORD.TOKEN_HERE" (leave the quotation marks). You can find out how to get your Discord token here.
WARNING: Never share your token with anyone or paste it into applications, websites, or bookmarklets you don't trust (you can see from that this one is completely harmless and only uses it for accessing Discord API)!
Tip: You can also use a bot's token to send a message as a bot, just add "Bot" before the token (example: "Bot xadafdfda.vcnm65cv_awaeou").
- Edit the if condition. First, turn on developer mode. Change all the text within quotes containing "Change this to a username or channel name" to a username or channel name you want to contact (you can shorten it to whatever you want to make it easier to type/remember). Then copy the channel ID (Not to be confused with a user ID. Even when DMing people, you have to use the DM's channel ID.) and put in it the second area containing "it's-id-here". Make as many name-ID pairs as you want for all the people or channels you want to quickly contact via this bookmarklet.
javascript: (function () {
const token = "PUT.YOUR_DISCORD.TOKEN_HERE";
const username = prompt("Enter username or channel:");
const users = JSON.parse(`{"Change this to a username or channel name":"it's-id-here","Change this to a username or channel name":"it's-id-here"}`);
if (users[username]) {
const message = prompt("Enter message:");
sendMessage();
async function sendMessage() {
const response = await fetch(`https://discord.com/api/v10/channels/${users[username]}/messages`, {
method: 'post',
body: JSON.stringify({
content: message
}),
headers: {
"Content-Type": "application/json",
"Authorization": token
}
});
const data = await response.json();
if (response.status !== 200) {
alert(`An error has occured! Error: ${response.status}`)
} else {
alert("Message sent!")
};
}
} else {
alert("User not found! Make sure you typed it correctly.")
}
})();
#discord #message #utility #performance | v2.2.0 | ⬆ |
---|
With this bookmarklet, you can send messages to people without booting up Discord (which is slow sometimes). There are two setup steps to get this working.
- Put your token into the area in the below code marked "PUT_YOUR_DISCORD_TOKEN_HERE" (leave the quotation marks). You can find out how to get your Discord token here.
WARNING: Never share your token with anyone or paste it into applications, websites, or bookmarklets you don't trust (you can see from that this one is completely harmless and only uses it for accessing Discord API)!
Tip: You can also use a bot's token to send a message as a bot, just add "Bot" before the token (example: "Bot xadafdfda.vcnm65cv_awaeou").
- Edit the if condition. First, turn on developer mode. Change all the text within quotes containing "Change this to a username or channel name" to a username or channel name you want to contact. Then copy the channel ID (Not to be confused with a user ID. Even when DMing people, you have to use the DM's channel ID.) and put in it the second area containing "it's-id-here". Make as many name-ID pairs as you want for all the people or channels you want to quickly contact via this bookmarklet.
Make sure to also change the default channel. Its format is slightly different, so make sure you only change the things marked "Choose Your Default Channel" and "default-id-here". Repeat it in the standard format, too.
Code in expandable area
javascript: (function () {
javascript: (function () {
const channels = JSON.parse(`{ "Default":{"Name":"Choose Your Default Channel","ID":"default-id-here"}, "Change this to a username": "it's id here", "Change this to a username": "it's id here"}`);
const token = "PUT_YOUR_DISCORD_TOKEN_HERE";
discordMessenger = document.createElement('div');
discordMessenger.id = 'discord-messenger';
discordMessenger.className = 'discord-messenger-class';
discordMessenger.innerHTML =
`<style>
.discord-messenger-class, discord-messenger-class * {
all: unset;
position: fixed;
inset: 10%;
z-index: 6000;
background-color: #313338;
}
.header {
width: 100%;
height: 48px;
padding: 8px;
box-sizing: border-box;
box-shadow: 0px 3px 2px -2px #1d1d1d;
position: relative;
display: flex;
flex-direction: column;
justify-content: center;
}
.channel-name {
font-size: 16px;
color: #f2f3f5;
text-align: left;
padding: 10px;
margin: 24px;
font-weight: 600;
font-family: 'Noto Sans', 'Helvetica Neue', 'Helvetica', 'Arial', 'sans-serif';
line-height: 20px;
cursor: pointer;
width: fit-content;
position: absolute
}
.channel-name:hover {
color: #b5bac1;
}
.channels-container {
background-color: #111214;;
min-width: 20%;
left: 37px;
top: 37px;
position: absolute;
border-radius: 3px;
z-index: 8000;
}
.channel-option {
font-size: 16px;
color: #b5bac1;
text-align: left;
font-weight: 600;
font-family: 'Noto Sans', 'Helvetica Neue', 'Helvetica', 'Arial', 'sans-serif';
line-height: 20px;
background-color: #111214;
padding: 3px;
margin: 3px;
border-radius: 3px;
display: block;
cursor: pointer;
}
.channel-option:hover {
background-color: #4752c4;
color: #ffffff;
}
.close-button {
fill: #b5bac1;
}
.close-button:hover {
fill: #DADCE0;
}
.msg-area {
border-radius: 5px;
position: absolute;
bottom: 11px;
width: 98%;
left: 1%;
background-color: #383a40;
overflow-wrap: break-word;
padding-bottom: 11px;
padding-top: 11px;
white-space: break-spaces !important;
text-align: left;
height: 32px;
}
.msg-input {
width: 100%;
display: block;
padding: 10px;
font-size: 14px;
margin: 50px auto;
border-radius: 6px;
border: 0;
outline: 0;
font-family: "gg sans", "Noto Sans", "Helvetica Neue", Helvetica, Arial, sans-serif;
font-weight: 400;
z-index: 7000;
bottom: -51px;
position: absolute;
background-color: #383a40;
color: #dbdee1;
resize: none;
}
.send-button {
bottom: 8px;
position: absolute;
z-index: 8001;
right: 10px;
fill: #4e5058;
cursor: not-allowed;
transition: fill .4s;
}
.content-true {
fill: #949cf7;
cursor: pointer;
}
.content-true:hover {
fill: #7971F3;
}
.messages-container {
overflow: hidden;
overflow-y: auto;
position: absolute;
top: 48px;
bottom: 65px;
left: 1%;
}
.message {
color: #2e3035;
}
.avatar {
}
.message-header {
margin: 0;
}
.username {
color: #f2f3f5;
font-size: 1rem;
font-weight: 500;
line-height: 1.375rem;
display: inline;
margin-right: .25rem;
font-family: "gg sans", "Noto Sans", "Helvetica Neue", Helvetica, Arial, sans-serif;
}
.timestamp {
color: #949ba4;
font-size: .75rem;
font-weight: 500;
line-height: 1.375rem;
display: inline;
margin-left: .25rem;
font-family: "gg sans", "Noto Sans", "Helvetica Neue", Helvetica, Arial, sans-serif;
}
.content {
color: #dbdee1;
font-size: 1rem;
line-height: 1.375rem;
font-family: "gg sans", "Noto Sans", "Helvetica Neue", Helvetica, Arial, sans-serif;
margin: 0;
margin-bottom: 1rem;
}
</style>
<div class="header">
<svg padding="10px" width="24" height="24" viewBox="0 0 24 24" x="0" y="0" aria-hidden="true" role="img">
<path fill="#80848e" fill-rule="evenodd" clip-rule="evenodd" d="M5.88657 21C5.57547 21 5.3399 20.7189 5.39427 20.4126L6.00001 17H2.59511C2.28449 17 2.04905 16.7198 2.10259 16.4138L2.27759 15.4138C2.31946 15.1746 2.52722 15 2.77011 15H6.35001L7.41001 9H4.00511C3.69449 9 3.45905 8.71977 3.51259 8.41381L3.68759 7.41381C3.72946 7.17456 3.93722 7 4.18011 7H7.76001L8.39677 3.41262C8.43914 3.17391 8.64664 3 8.88907 3H9.87344C10.1845 3 10.4201 3.28107 10.3657 3.58738L9.76001 7H15.76L16.3968 3.41262C16.4391 3.17391 16.6466 3 16.8891 3H17.8734C18.1845 3 18.4201 3.28107 18.3657 3.58738L17.76 7H21.1649C21.4755 7 21.711 7.28023 21.6574 7.58619L21.4824 8.58619C21.4406 8.82544 21.2328 9 20.9899 9H17.41L16.35 15H19.7549C20.0655 15 20.301 15.2802 20.2474 15.5862L20.0724 16.5862C20.0306 16.8254 19.8228 17 19.5799 17H16L15.3632 20.5874C15.3209 20.8261 15.1134 21 14.8709 21H13.8866C13.5755 21 13.3399 20.7189 13.3943 20.4126L14 17H8.00001L7.36325 20.5874C7.32088 20.8261 7.11337 21 6.87094 21H5.88657ZM9.41045 9L8.35045 15H14.3504L15.4104 9H9.41045Z"></path>
</svg>
<h1 id="channel-name" class="channel-name">${channels.Default.Name}</h1>
<svg style="right:2%;position:absolute;" aria-hidden="true" role="img" width="24" height="24" viewBox="0 0 14 14">
<path class="close-button" id="close-button" fill="#b5bac1" d="M7.02799 0.333252C3.346 0.333252 0.361328 3.31792 0.361328 6.99992C0.361328 10.6819 3.346 13.6666 7.02799 13.6666C10.71 13.6666 13.6947 10.6819 13.6947 6.99992C13.6947 3.31792 10.7093 0.333252 7.02799 0.333252ZM10.166 9.19525L9.22333 10.1379L7.02799 7.94325L4.83266 10.1379L3.89 9.19525L6.08466 6.99992L3.88933 4.80459L4.832 3.86259L7.02733 6.05792L9.22266 3.86259L10.1653 4.80459L7.97066 6.99992L10.166 9.19525Z"></path>
</svg>
</div>
<div id="channels-container" class="channels-container"></div>
<div id="messages-container" class="messages-container"></div>
<div class="msg-area">
<svg aria-hidden="true" role="img" id="send-button" class="send-button" width="16" height="16" viewBox="0 0 16 16">
<path d="M8.2738 8.49222L1.99997 9.09877L0.349029 14.3788C0.250591 14.691 0.347154 15.0322 0.595581 15.246C0.843069 15.4597 1.19464 15.5047 1.48903 15.3613L15.2384 8.7032C15.5075 8.57195 15.6781 8.29914 15.6781 8.00007C15.6781 7.70101 15.5074 7.4282 15.2384 7.29694L1.49839 0.634063C1.20401 0.490625 0.852453 0.535625 0.604941 0.749376C0.356493 0.963128 0.259941 1.30344 0.358389 1.61563L2.00932 6.89563L8.27093 7.50312C8.52405 7.52843 8.71718 7.74125 8.71718 7.99531C8.71718 8.24938 8.52406 8.46218 8.27093 8.4875L8.2738 8.49222Z"></path>
</svg>
<textarea class='msg-input auto-expand' rows='1' data-min-rows='1' style="overflow:hidden;" id="msg-input" placeholder='Message #${channels.Default.Name}' autofocus></textarea>
</div>`;
document.body.appendChild(discordMessenger);
let targetID = channels.Default.ID;
getMessages();
document.getElementById("msg-input").addEventListener('input', autoResize, false);
document.getElementById("msg-input").addEventListener('input', textAreaStatus);
const textArea = document.getElementById('msg-input');
function autoResize() {
textArea.style.height = 'auto';
textArea.style.height = textArea.scrollHeight + 'px';
}
textArea.addEventListener('keydown', async (event) => {
if (event.key === 'Enter') {
if (event.shiftKey) {
textArea.style.height = parseInt(textArea.style.height) + 17 + 'px';
const start = textArea.selectionStart;
const end = textArea.selectionEnd;
const value = textArea.value;
textArea.value = value.substring(0, start) + '\n' + value.substring(end);
textArea.selectionStart = textArea.selectionEnd = start + 1;
event.preventDefault();
textAreaStatus()
} else {
event.preventDefault();
sendMessage();
textAreaStatus();
}
}
});
const sendButton = document.getElementById("send-button");
function textAreaStatus() {
if (textArea.value === "") {
sendButton.classList.remove("content-true");
} else {
sendButton.classList.add("content-true");
}
}
async function sendMessage() {
const message = textArea.value;
if (message) {
try {
const response = await fetch(`https://discord.com/api/v10/channels/${targetID}/messages`, {
method: 'post',
body: JSON.stringify({
content: message
}),
headers: {
"Content-Type": "application/json",
"Authorization": token
}
});
const data = await response.json();
console.log(data);
textArea.value = '';
textArea.rows = 1;
populateMessage(data, "single");
} catch (error) {
console.error(error);
}
}
}
function close() {
document.getElementById('discord-messenger').remove();
}
const channelsContainer = document.getElementById('channels-container');
const channelName = document.getElementById('channel-name');
discordMessenger.addEventListener('click', (event) => {
if (event.target === channelName) {
if (channelsContainer.children.length === 0) {
for (const [key, value] of Object.entries(channels).slice(1)) {
const channel = document.createElement('channel');
channel.id = value;
channel.className = 'channel-option';
channel.innerHTML = key;
channel.addEventListener('click', () => {
targetID = channel.id;
let targetName = channel.innerHTML;
channelName.innerHTML = targetName;
textArea.placeholder = 'Message #' + targetName;
document.querySelectorAll('.channel-option').forEach(e => e.remove());
getMessages();
});
channelsContainer.appendChild(channel);
}
} else {
document.querySelectorAll('.channel-option').forEach(e => e.remove());
}
} else {
document.querySelectorAll('.channel-option').forEach(e => e.remove());
}
});
const messagesContainer = document.getElementById('messages-container');
async function getMessages() {
try {
const response = await fetch(`https://discord.com/api/v10/channels/${targetID}/messages`, {
method: 'get',
headers: {
"Content-Type": "application/json",
"Authorization": token
}
});
const data = await response.json();
console.log(data);
messagesContainer.innerHTML = '';
for (let i = 0; i < data.length; i++) {
populateMessage(data[i], "multiple");
}
} catch (error) {
console.error(error);
}
}
function populateMessage(data, type) {
const messagesContainer = document.getElementById('messages-container');
const newMessage = document.createElement('span');
newMessage.class = "message-container";
newMessage.id = `${data.id}`;
newMessage.classList = "message";
newMessage.innerHTML = `<h3 class="message-header"><p class="username">${data.author.global_name}</p><time class="timestamp" datetime="${data.timestamp}">${data.timestamp}</time></h3><p class="content">${data.content}</p>`;
if (type == "single") {
messagesContainer.appendChild(newMessage);
} else if (type == "multiple") {
messagesContainer.insertBefore(newMessage, messagesContainer.firstChild);
}
messagesContainer.scrollTo({ left: 0, top: messagesContainer.scrollHeight });
}
document.getElementById('close-button').addEventListener('click', close);
document.getElementById('send-button').addEventListener('click', sendMessage);
}
})();
#dev #utility #editor | v1.0.0 | ⬆ |
---|
You can move any element you want around the web page! Really useful for web developers!
javascript: var b = X = Y = T = L = 0;
document.addEventListener("click", function(a) {
a.preventDefault()
}, !0);
document.addEventListener("mousedown", c);
document.addEventListener("touchstart", c);
function c(a) {
a.preventDefault();
a.target !== document.documentElement && a.target !== document.body && (b = Date.now(), a.target.setAttribute("data-drag", b), a.target.style.position = "relative", T = a.target.style.top.split("px")[0] || 0, L = a.target.style.left.split("px")[0] || 0);
X = a.clientX || a.touches[0].clientX;
Y = a.clientY || a.touches[0].clientY
}
document.addEventListener("mousemove", d);
document.addEventListener("touchmove", d);
function d(a) {
if ("" !== b) {
var e = document.querySelector('[data-drag="' + b + '"]');
e.style.top = parseInt(T) + parseInt((a.clientY || a.touches[0].clientY) - Y) + "px";
e.style.left = parseInt(L) + parseInt((a.clientX || a.touches[0].clientX) - X) + "px"
}
}
document.addEventListener("mouseup", f);
document.addEventListener("touchend", f);
function f() {
b = ""
}
document.addEventListener("mouseover", g);
function g(a) {
a.target.style.cursor = "move";
a.target.style.boxShadow = "inset lime 0 0 1px,lime 0 0 1px"
}
document.addEventListener("mouseout", h);
function h(a) {
a.target.style.cursor = a.target.style.boxShadow = ""
};
#utility | v1.0.0 | ⬆ |
---|
Instantly scroll to the bottom of the page.
javascript: window.scrollTo(0, document.body.scrollHeight || document.documentElement.scrollHeight);
#utility | v1.0.0 | ⬆ |
---|
Like the previous one, it scrolls to the bottom of the page, except it never stops scrolling. Useful for when new content is continuously added to the website.
javascript: setInterval(function() {
window.scrollTo(0, document.body.scrollHeight);
}, 100);
window.stop();
#utility #editor | v1.0.1 | ⬆ |
---|
Change the title to just the root page's name, by setting the page's title to the text in the title that comes after a "|", "-", or "•".
javascript: (function() {
var title = document.title.trim();
var match = title.match(/.*[\|\-%E2%80%A2]\s*(.*)$/);
if (match && match[1]) {
var newTitle = match[1].trim();
document.title = newTitle;
} else {
var newTitle = prompt("Enter a new title:", title);
if (newTitle !== null) {
document.title = newTitle;
}
}
})();
#utility #anonymous #social | v1.0.0 | ⬆ |
---|
When viewing a profile on Instagram, click this bookmarklet to view the profile on InstaNavigation.com/.
javascript: (function() {
var path = window.location.href.replace(/^https?:\/\/[^/]+/, '');
var parts = path.split('/').filter(Boolean);
if (parts.length > 0) {
window.open("https://instanavigation.com/user-profile/" + parts[0], '_blank');
} else {
alert('Could not extract username from the URL.');
}
})();