Skip to content

Instantly share code, notes, and snippets.

@kvdogan
Forked from kepano/obsidian-web-clipper.js
Last active March 23, 2023 10:49
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save kvdogan/cf55f62781419037a572773000e2858c to your computer and use it in GitHub Desktop.
Save kvdogan/cf55f62781419037a572773000e2858c to your computer and use it in GitHub Desktop.
Obsidian Web Clipper Bookmarklet to save articles and pages from the web (for Safari, Chrome, Firefox, and mobile browsers)

Modified and improved by @kvdogan in terms of:

  1. Parsing long page was silently failing due to url character limitation, that issue has been fixed by utilizing Advanced URI plugin.
  2. Added backlink to the site that has been parsed together with parsed character quantity.

By @kepano

🎉 Support my work at buymeacoffee.com/kepano

Demo

You can find a demo of this bookmarklet on YouTube

Installation

Create a new bookmark in your browser, then copy/paste the minified code below into the URL field.

You can customize the output using the optional variables at the top, and the template at the bottom. The default template is designed for use with the Dataview plugin. If you make changes I recommend using Bookmarklet Maker to minify and URI encode the bookmarklet.

Usage

By default, clicking the bookmarklet creates a new Obsidian file from the main body of the article (similar to Readability view). Alternatively you can choose to create a file from a selection, by either selecting all (CMD+A), or just a portion of the page.

Any images in the content will be embedded as external references. If you want to download images locally you can use the Local Images plugin which allows you to download images for a note.

Troubleshooting

This bookmarklet may not work on all websites. If you run into issues, you can also try the MarkDownload browser extension which provides similar functionality. You can troubleshoot issues by opening the Developer Console in your browser and checking if any errors appear when you click the bookmarklet. The most common error is that a website or the browser itself is blocking third party code execution. Unfortunately there is no good solve for that yet.

javascript: Promise.all([import('https://unpkg.com/turndown@6.0.0?module'), import('https://unpkg.com/@tehshrike/readability@0.2.0'),]).then(async ([{
default: Turndown
}, {
default: Readability
}]) => {
/* Optional vault name */
const vault = "";
/* Optional folder name such as "Clippings/" */
const folder = "";
/* Optional tags */
let tags = "clippings";
/* parse and lightly clean the site's meta keywords content into tags, if present */
if (document.querySelector('meta[name="keywords" i]')) {
var keywords = document.querySelector('meta[name="keywords" i]').getAttribute('content').split(',');
keywords.forEach(function (keyword) {
let tag = ' ' + keyword.split(' ').join('');
tags += tag;
});
}
function getSelectionHtml() {
var html = "";
if (typeof window.getSelection != "undefined") {
var sel = window.getSelection();
if (sel.rangeCount) {
var container = document.createElement("div");
for (var i = 0, len = sel.rangeCount; i < len; ++i) {
container.appendChild(sel.getRangeAt(i).cloneContents());
}
html = container.innerHTML;
}
} else if (typeof document.selection != "undefined") {
if (document.selection.type == "Text") {
html = document.selection.createRange().htmlText;
}
}
return html;
}
const selection = getSelectionHtml();
const {
title,
byline,
content
} = new Readability(document.cloneNode(true)).parse();
function getFileName(fileName) {
var userAgent = window.navigator.userAgent,
platform = window.navigator.platform,
windowsPlatforms = ['Win32', 'Win64', 'Windows', 'WinCE'];
if (windowsPlatforms.indexOf(platform) !== -1) {
fileName = fileName.replace(':', '').replace(/[/\\?%*|"<>]/g, '-');
} else {
fileName = fileName.replace(':', '').replace(/\//g, '-').replace(/\\/g, '-');
}
return fileName;
}
const fileName = getFileName(title);
if (selection) {
var markdownify = selection;
} else {
var markdownify = content;
}
if (vault) {
var vaultName = '&vault=' + encodeURIComponent(`${vault}`);
} else {
var vaultName = '';
}
const markdownBody = new Turndown({
headingStyle: 'atx',
hr: '~~~',
bulletListMarker: '-',
codeBlockStyle: 'fenced',
emDelimiter: '*',
}).turndown(markdownify);
var date = new Date();
function convertDate(date) {
var yyyy = date.getFullYear().toString();
var mm = (date.getMonth() + 1).toString();
var dd = date.getDate().toString();
var mmChars = mm.split('');
var ddChars = dd.split('');
return yyyy + '-' + (mmChars[1] ? mm : "0" + mmChars[0]) + '-' + (ddChars[1] ? dd : "0" + ddChars[0]);
}
const today = convertDate(date);
/* YAML front matter as tags render cleaner with special chars */
const fileContent =
"---\n"
+ "author: " + byline + "\n"
+ "title: [" + title + "]\n"
+ "source: " + document.URL + "\n"
+ "clip date: " + today + "\n"
+ "published: \n\n"
+ "tags: [" + tags + "]\n"
+ "---\n\n"
+ markdownBody
+ "\n\n---\n" + String(markdownBody.length) + " chars clipped from "
+ "[" + title + "](" + document.URL + ")\n";
// This function must be called in a visible page, such as a browserAction popup
// or a content script. Calling it in a background page has no effect!
async function copyContent() {
try {
await navigator.clipboard.writeText(fileContent);
} catch (err) {
console.error('Failed to copy: ', err);
}
}
copyContent();
document.location.href = "obsidian://advanced-uri?"
+ "vault=" + vaultName
+ "&clipboard=true"
+ "&mode=new"
+ "&filepath=" + encodeURIComponent(folder + fileName);
})
javascript:(function()%7Bjavascript%3A%20Promise.all(%5Bimport('https%3A%2F%2Funpkg.com%2Fturndown%406.0.0%3Fmodule')%2C%20import('https%3A%2F%2Funpkg.com%2F%40tehshrike%2Freadability%400.2.0')%2C%5D).then(async%20(%5B%7B%0A%20%20%20%20default%3A%20Turndown%0A%7D%2C%20%7B%0A%20%20%20%20default%3A%20Readability%0A%7D%5D)%20%3D%3E%20%7B%0A%0A%20%20%20%20%2F*%20Optional%20vault%20name%20*%2F%0A%20%20%20%20const%20vault%20%3D%20%22%22%3B%0A%0A%20%20%20%20%2F*%20Optional%20folder%20name%20such%20as%20%22Clippings%2F%22%20*%2F%0A%20%20%20%20const%20folder%20%3D%20%22%22%3B%0A%0A%20%20%20%20%2F*%20Optional%20tags%20%20*%2F%0A%20%20%20%20let%20tags%20%3D%20%22clippings%22%3B%0A%0A%20%20%20%20%2F*%20parse%20and%20lightly%20clean%20the%20site's%20meta%20keywords%20content%20into%20tags%2C%20if%20present%20*%2F%0A%20%20%20%20if%20(document.querySelector('meta%5Bname%3D%22keywords%22%20i%5D'))%20%7B%0A%20%20%20%20%20%20%20%20var%20keywords%20%3D%20document.querySelector('meta%5Bname%3D%22keywords%22%20i%5D').getAttribute('content').split('%2C')%3B%0A%0A%20%20%20%20%20%20%20%20keywords.forEach(function%20(keyword)%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20let%20tag%20%3D%20'%20'%20%2B%20keyword.split('%20').join('')%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20tags%20%2B%3D%20tag%3B%0A%20%20%20%20%20%20%20%20%7D)%3B%0A%20%20%20%20%7D%0A%0A%20%20%20%20function%20getSelectionHtml()%20%7B%0A%20%20%20%20%20%20%20%20var%20html%20%3D%20%22%22%3B%0A%20%20%20%20%20%20%20%20if%20(typeof%20window.getSelection%20!%3D%20%22undefined%22)%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20var%20sel%20%3D%20window.getSelection()%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20if%20(sel.rangeCount)%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20var%20container%20%3D%20document.createElement(%22div%22)%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20for%20(var%20i%20%3D%200%2C%20len%20%3D%20sel.rangeCount%3B%20i%20%3C%20len%3B%20%2B%2Bi)%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20container.appendChild(sel.getRangeAt(i).cloneContents())%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%7D%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20html%20%3D%20container.innerHTML%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20%7D%0A%20%20%20%20%20%20%20%20%7D%20else%20if%20(typeof%20document.selection%20!%3D%20%22undefined%22)%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20if%20(document.selection.type%20%3D%3D%20%22Text%22)%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20html%20%3D%20document.selection.createRange().htmlText%3B%0A%20%20%20%20%20%20%20%20%20%20%20%20%7D%0A%20%20%20%20%20%20%20%20%7D%0A%20%20%20%20%20%20%20%20return%20html%3B%0A%20%20%20%20%7D%0A%0A%20%20%20%20const%20selection%20%3D%20getSelectionHtml()%3B%0A%0A%20%20%20%20const%20%7B%0A%20%20%20%20%20%20%20%20title%2C%0A%20%20%20%20%20%20%20%20byline%2C%0A%20%20%20%20%20%20%20%20content%0A%20%20%20%20%7D%20%3D%20new%20Readability(document.cloneNode(true)).parse()%3B%0A%0A%20%20%20%20function%20getFileName(fileName)%20%7B%0A%20%20%20%20%20%20%20%20var%20userAgent%20%3D%20window.navigator.userAgent%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20platform%20%3D%20window.navigator.platform%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20windowsPlatforms%20%3D%20%5B'Win32'%2C%20'Win64'%2C%20'Windows'%2C%20'WinCE'%5D%3B%0A%0A%20%20%20%20%20%20%20%20if%20(windowsPlatforms.indexOf(platform)%20!%3D%3D%20-1)%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20fileName%20%3D%20fileName.replace('%3A'%2C%20'').replace(%2F%5B%2F%5C%5C%3F%25*%7C%22%3C%3E%5D%2Fg%2C%20'-')%3B%0A%20%20%20%20%20%20%20%20%7D%20else%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20fileName%20%3D%20fileName.replace('%3A'%2C%20'').replace(%2F%5C%2F%2Fg%2C%20'-').replace(%2F%5C%5C%2Fg%2C%20'-')%3B%0A%20%20%20%20%20%20%20%20%7D%0A%20%20%20%20%20%20%20%20return%20fileName%3B%0A%20%20%20%20%7D%0A%20%20%20%20const%20fileName%20%3D%20getFileName(title)%3B%0A%0A%20%20%20%20if%20(selection)%20%7B%0A%20%20%20%20%20%20%20%20var%20markdownify%20%3D%20selection%3B%0A%20%20%20%20%7D%20else%20%7B%0A%20%20%20%20%20%20%20%20var%20markdownify%20%3D%20content%3B%0A%20%20%20%20%7D%0A%0A%20%20%20%20if%20(vault)%20%7B%0A%20%20%20%20%20%20%20%20var%20vaultName%20%3D%20'%26vault%3D'%20%2B%20encodeURIComponent(%60%24%7Bvault%7D%60)%3B%0A%20%20%20%20%7D%20else%20%7B%0A%20%20%20%20%20%20%20%20var%20vaultName%20%3D%20''%3B%0A%20%20%20%20%7D%0A%0A%20%20%20%20const%20markdownBody%20%3D%20new%20Turndown(%7B%0A%20%20%20%20%20%20%20%20headingStyle%3A%20'atx'%2C%0A%20%20%20%20%20%20%20%20hr%3A%20'~~~'%2C%0A%20%20%20%20%20%20%20%20bulletListMarker%3A%20'-'%2C%0A%20%20%20%20%20%20%20%20codeBlockStyle%3A%20'fenced'%2C%0A%20%20%20%20%20%20%20%20emDelimiter%3A%20'*'%2C%0A%20%20%20%20%7D).turndown(markdownify)%3B%0A%0A%20%20%20%20var%20date%20%3D%20new%20Date()%3B%0A%0A%20%20%20%20function%20convertDate(date)%20%7B%0A%20%20%20%20%20%20%20%20var%20yyyy%20%3D%20date.getFullYear().toString()%3B%0A%20%20%20%20%20%20%20%20var%20mm%20%3D%20(date.getMonth()%20%2B%201).toString()%3B%0A%20%20%20%20%20%20%20%20var%20dd%20%3D%20date.getDate().toString()%3B%0A%20%20%20%20%20%20%20%20var%20mmChars%20%3D%20mm.split('')%3B%0A%20%20%20%20%20%20%20%20var%20ddChars%20%3D%20dd.split('')%3B%0A%20%20%20%20%20%20%20%20return%20yyyy%20%2B%20'-'%20%2B%20(mmChars%5B1%5D%20%3F%20mm%20%3A%20%220%22%20%2B%20mmChars%5B0%5D)%20%2B%20'-'%20%2B%20(ddChars%5B1%5D%20%3F%20dd%20%3A%20%220%22%20%2B%20ddChars%5B0%5D)%3B%0A%20%20%20%20%7D%0A%0A%20%20%20%20const%20today%20%3D%20convertDate(date)%3B%0A%0A%20%20%20%20%2F*%20YAML%20front%20matter%20as%20tags%20render%20cleaner%20with%20special%20chars%20%20*%2F%0A%20%20%20%20const%20fileContent%20%3D%0A%20%20%20%20%20%20%20%20%22---%5Cn%22%0A%20%20%20%20%20%20%20%20%2B%20%22author%3A%20%20%20%20%22%20%2B%20byline%20%2B%20%22%5Cn%22%0A%20%20%20%20%20%20%20%20%2B%20%22title%3A%20%20%20%20%20%5B%22%20%2B%20title%20%2B%20%22%5D%5Cn%22%0A%20%20%20%20%20%20%20%20%2B%20%22source%3A%20%20%20%20%22%20%2B%20document.URL%20%2B%20%22%5Cn%22%0A%20%20%20%20%20%20%20%20%2B%20%22clip%20date%3A%20%22%20%2B%20today%20%2B%20%22%5Cn%22%0A%20%20%20%20%20%20%20%20%2B%20%22published%3A%20%5Cn%5Cn%22%0A%20%20%20%20%20%20%20%20%2B%20%22tags%3A%20%20%20%20%20%20%5B%22%20%2B%20tags%20%2B%20%22%5D%5Cn%22%0A%20%20%20%20%20%20%20%20%2B%20%22---%5Cn%5Cn%22%0A%20%20%20%20%20%20%20%20%2B%20markdownBody%0A%20%20%20%20%20%20%20%20%2B%20%22%5Cn%5Cn---%5Cn%22%20%2B%20String(markdownBody.length)%20%2B%20%22%20chars%20clipped%20from%20%22%0A%20%20%20%20%20%20%20%20%2B%20%22%5B%22%20%2B%20title%20%2B%20%22%5D(%22%20%2B%20document.URL%20%2B%20%22)%5Cn%22%3B%0A%0A%0A%20%20%20%20%2F%2F%20This%20function%20must%20be%20called%20in%20a%20visible%20page%2C%20such%20as%20a%20browserAction%20popup%0A%20%20%20%20%2F%2F%20or%20a%20content%20script.%20Calling%20it%20in%20a%20background%20page%20has%20no%20effect!%0A%20%20%20%20async%20function%20copyContent()%20%7B%0A%20%20%20%20%20%20%20%20try%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20await%20navigator.clipboard.writeText(fileContent)%3B%0A%20%20%20%20%20%20%20%20%7D%20catch%20(err)%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20console.error('Failed%20to%20copy%3A%20'%2C%20err)%3B%0A%20%20%20%20%20%20%20%20%7D%0A%20%20%20%20%7D%0A%0A%20%20%20%20copyContent()%3B%0A%0A%20%20%20%20document.location.href%20%3D%20%22obsidian%3A%2F%2Fadvanced-uri%3F%22%0A%20%20%20%20%20%20%20%20%2B%20%22vault%3D%22%20%2B%20vaultName%0A%20%20%20%20%20%20%20%20%2B%20%22%26clipboard%3Dtrue%22%0A%20%20%20%20%20%20%20%20%2B%20%22%26mode%3Dnew%22%0A%20%20%20%20%20%20%20%20%2B%20%22%26filepath%3D%22%20%2B%20encodeURIComponent(folder%20%2B%20fileName)%3B%0A%0A%7D)%7D)()%3B
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment