JavaScript: Invert color on all elements of a page
Solution 1
My solution seems to work only for Chrome right now, but it inverts everything (including images and iframes) as seen here:
Also it does not make use of external libraries and is very simple: adding a -webkit-filter: invert(100%)
to the html
-selector.
javascript: (
function () {
// the css we are going to inject
var css = 'html {-webkit-filter: invert(100%);' +
'-moz-filter: invert(100%);' +
'-o-filter: invert(100%);' +
'-ms-filter: invert(100%); }',
head = document.getElementsByTagName('head')[0],
style = document.createElement('style');
// a hack, so you can "invert back" clicking the bookmarklet again
if (!window.counter) { window.counter = 1;} else { window.counter ++;
if (window.counter % 2 == 0) { var css ='html {-webkit-filter: invert(0%); -moz-filter: invert(0%); -o-filter: invert(0%); -ms-filter: invert(0%); }'}
};
style.type = 'text/css';
if (style.styleSheet){
style.styleSheet.cssText = css;
} else {
style.appendChild(document.createTextNode(css));
}
//injecting the css to the head
head.appendChild(style);
}());
Here's the fiddle: http://jsfiddle.net/nikita_turing/jVKw6/3/ with the bookmarklet included. If someone has an idea of how to make it work for Firefox (SVG-Filters?), go ahead!
Solution 2
First things first, grab the awesome RGBColor
class here.
Here goes:
jsFiddle example
//set up color properties to iterate through
var colorProperties = ['color', 'background-color'];
//iterate through every element in reverse order...
$("*").get().reverse().each(function() {
var color = null;
for (var prop in colorProperties) {
prop = colorProperties[prop];
//if we can't find this property or it's null, continue
if (!$(this).css(prop)) continue;
//create RGBColor object
color = new RGBColor($(this).css(prop));
if (color.ok) {
//good to go, let's build up this RGB baby!
//subtract each color component from 255
$(this).css(prop, 'rgb(' + (255 - color.r) + ', ' + (255 - color.g) + ', ' + (255 - color.b) + ')');
}
color = null; //some cleanup
}
});
Screenshot:
EDIT: Here's a bookmarklet you can now copy-paste into your browser (http://jsfiddle.net/F7HqS/1/)
javascript:function load_script(src,callback){var s=document.createElement('script');s.src=src;s.onload=callback;document.getElementsByTagName('head')[0].appendChild(s);}function invertColors(){var colorProperties=['color','background-color'];$('*').each(function(){var color=null;for(var prop in colorProperties){prop=colorProperties[prop];if(!$(this).css(prop))continue;color=new RGBColor($(this).css(prop));if(color.ok){$(this).css(prop,'rgb('+(255-color.r)+','+(255-color.g)+','+(255-color.b)+')');}color=null;}});}load_script('http://www.phpied.com/files/rgbcolor/rgbcolor.js',function(){if(!window.jQuery)load_script('https://ajax.googleapis.com/ajax/libs/jquery/1.4.4/jquery.min.js',invertColors);else invertColors();});
Solution 3
I cleaned up the comments from one of the answers (by leosok) above, so it will work as a bookmarklet in chrome. Note that this solution is more efficient than the current highest-point here, plus it works even if the html changes after you run the script.
javascript:(function () {
var css = 'html {-webkit-filter: invert(100%);' + '-moz-filter: invert(100%);' + '-o-filter: invert(100%);' + '-ms-filter: invert(100%); }';
var head = document.getElementsByTagName('head')[0];
var style = document.createElement('style');
if (!window.counter) {
window.counter = 1;
} else {
window.counter++;
if (window.counter % 2 == 0) {
var css = 'html {-webkit-filter: invert(0%); -moz-filter: invert(0%); -o-filter: invert(0%); -ms-filter: invert(0%); }'
}
}
style.type = 'text/css';
if (style.styleSheet) {
style.styleSheet.cssText = css;
} else {
style.appendChild(document.createTextNode(css));
}
head.appendChild(style);
}());
One line for bookmarklet. create a bookmark, then edit the url to this:
javascript:(function () { var css = 'html {-webkit-filter: invert(100%);' + '-moz-filter: invert(100%);' + '-o-filter: invert(100%);' + '-ms-filter: invert(100%); }'; var head = document.getElementsByTagName('head')[0]; var style = document.createElement('style'); if (!window.counter) { window.counter = 1; } else { window.counter++; if (window.counter % 2 == 0) { var css = 'html {-webkit-filter: invert(0%); -moz-filter: invert(0%); -o-filter: invert(0%); -ms-filter: invert(0%); }' } } style.type = 'text/css'; if (style.styleSheet) { style.styleSheet.cssText = css; } else { style.appendChild(document.createTextNode(css)); } head.appendChild(style); }());
Solution 4
The accepted answer is totally correct, with one minor flaw. Each time you toggle the invert it adds ANOTHER style tag to the head. Do this instead
// the css we are going to inject
let css = 'html {-webkit-filter: invert(100%);' +
'-moz-filter: invert(100%);' +
'-o-filter: invert(100%);' +
'-ms-filter: invert(100%); }';
let head = $('head')[0];
let invertStyle = $('#invert')[0];
if (invertStyle) {
head.removeChild(invertStyle);
} else {
let style = document.createElement('style');
style.type = 'text/css';
style.id = 'invert';
if (style.styleSheet){
style.styleSheet.cssText = css;
} else {
style.appendChild(document.createTextNode(css));
}
//injecting the css to the head
head.appendChild(style);
}
That way you simply remove the tag if you want to undo your invert. Works great!
Solution 5
I figured it would be fun to try inverting images. Didn't take long to find an appropriate Javascript library for image editing: http://www.pixastic.com/lib/
You probably can't load that whole library into a bookmarklet, but if you host it yourself you can add something like this to the end of the bookmarklet (after invertColors):
load_script('http://www.example.com/pixastic.invert.js', function () {$('img').each(function() {try{$(this).pixastic("invert");} catch(e) {}})})
I think it's worth noting that if your goal is to take a page with a white background and make it black (or vice versa), something simpler might be in order.
I just tried the bookmarklet from Jacob and compared it to a more naive version I found on the Google support forums: http://www.google.com/support/forum/p/Chrome/thread?tid=26affebdd0da12d9&hl=en
Jacob's invert seems to work less frequently and takes quite a bit longer on large pages. I think I'll end up using the naive version more frequently.
Related videos on Youtube
Comments
-
Muhd over 3 years
Note: I am keeping an up-to-date version of the bookmarklet in my question which works well and is based on Jacob's answer. If you are looking for a bookmarklet to use, use that one.See leosok's fantastic answer if you just want something amazing that works on chrome.I want to be able to invert the color of all the elements on a page with a JavaScript bookmarklet. I know that to invert a color you subtract each of the RGB hex values from 255(xFF), but beyond that I am unsure of how to proceed.
How can I accomplish this?
Using
jQuery
is acceptable, and it only needs to work on Chrome, although if it worked in Firefox that'd be a plus.This is excluding images - background, text and links colors should all be inverted. Basically anything that gets its color from CSS.
UPDATE Here is an updated bookmarklet that fixes the nested element issue and will work on a lot of different sites (including this one)
UPDATE2 Added some support for transparency, handling elements that have default background-color rgba(0, 0, 0, 0). More sites should be working now with the updated one.
javascript: (function ($) { function load_script(src, callback) { var s = document.createElement('script'); s.src = src; s.onload = callback; document.getElementsByTagName('head')[0].appendChild(s); } function invertElement() { var colorProperties = ['color', 'background-color']; var color = null; for (var prop in colorProperties) { prop = colorProperties[prop]; if (!$(this).css(prop)) continue; if ($(this).data(prop) != $(this).css(prop)) continue; if (($(this).css(prop) === 'rgba(0, 0, 0, 0)') || ($(this).css(prop) === 'transparent')) { if ($(this).is('body')) { $(this).css(prop, 'black'); continue; } else { continue; } } color = new RGBColor($(this).css(prop)); if (color.ok) { $(this).css(prop, 'rgb(' + (255 - color.r) + ',' + (255 - color.g) + ',' + (255 - color.b) + ')'); } color = null; } } function setColorData() { var colorProperties = ['color', 'background-color']; for (var prop in colorProperties) { prop = colorProperties[prop]; $(this).data(prop, $(this).css(prop)); } } function invertColors() { $(document).live('DOMNodeInserted', function(e) { var $toInvert = $(e.target).find('*').andSelf(); $toInvert.each(setColorData); $toInvert.each(invertElement); }); $('*').each(setColorData); $('*').each(invertElement); $('iframe').each(function () { $(this).contents().find('*').each(setColorData); $(this).contents().find('*').each(invertElement); }); } load_script('http://www.phpied.com/files/rgbcolor/rgbcolor.js', function () { if (!window.jQuery) load_script('https://ajax.googleapis.com/ajax/libs/jquery/1.7.1/jquery.min.js', invertColors); else invertColors(); }); })(jQuery);
Now works with most sites I've tried. Background images can pose a problem, however.
-
Muhd about 12 yearsApparently you can't inline comments in a bookmarklet since it gets converted into one line.
-
Muhd about 12 yearsI have removed the comment which was breaking the bookmarklet, and now just do
$('*')
for selecting all elements so that it now includes the<html>
element (which can apparently have a background color). -
ggorlen over 3 yearsIt's preferred to write a self-answer than to keep editing the original post like this. I realize this is a community wiki now, but it still makes of an awkward thread to get information from--answers should go into the, well, "answers" field so they can be voted upon and evaluated on their own merits.
-
-
Muhd over 13 years+1 This is very nice, but I am having trouble making it into a bookmarklet
-
Jacob Relkin over 13 years@Muhd See my updated answer, just grab that code and stick it in a browser's address bar with
javascript:
preceding it and voila! :) -
Muhd over 13 yearsThis is super cool! However, when used on this page, it isn't properly inverting the text of my question, or your answer. I wonder why that is.
-
Kranu over 13 years+1 I was just browsing, but this bookmarklet will come in handy :)
-
Jacob Relkin over 13 years@Muhd I think I might have to modify the
RGBColor
class for this one, apparently it doesn't support alpha. :( -
Muhd over 13 yearsYou know, I actually thought that alpha might be a major hurdle even before asking the question. Also, it might be nice if border colors were factored in, although they are a little more complicated since they can be top, bottom, left, and right. Some pages look bad without the borders inverted (like a google search result).
-
Admin over 13 yearsI thing it's StackOverflow Nightshow. well don't show this to jeffatwood. they thing that you make a another clone OF SO.
-
Muhd over 13 yearsA number of other sites have the problem as well, so if you need more references just look around. google search and google finance seem to have the problem.
-
Jacob Relkin over 13 years@Muhd I'm working on it. Hang on a bit - i'll let you know when it's done.
-
Muhd over 13 years@Jacob, I modified the code to handle iframe contents: jsfiddle.net/F7HqS/5 .
-
Jacob Relkin over 13 years@Muhd Cool. I'm almost done with my modifications to the
RGBColor
class. I'll integrate your changes in the finished package. -
Muhd over 13 years@Jacob, I have actually found a reason that a number of pages are not working properly. The text is being inverted multiple times: ScrnSht.com/cvkner This is because the text color will be inverted for an element, and then that element's child will invert the color back, and then THAT element's child will invert it again. When the CSS is custom defined for lowest level element in the stylesheet, or the number of inversions just happens to be odd, it works out to be the correct color. When there is an even number of generations from the last color definition the color is incorrect.
-
Jacob Relkin over 13 years@Muhd Thanks for that - I will take that into consideration.
-
Jacob Relkin over 13 years@Muhd See my latest question. stackoverflow.com/questions/5000108/…
-
Muhd over 13 yearsOne solution might be to store the color properties of each element prior to any inversion and then use that as a reference. If it changed, then it is inherited. Otherwise not. Perhaps less complicated than any solution you get for the above question.
-
Jacob Relkin about 13 years@Muhd Okay thanks. Will let you know when I get around to finishing this.
-
Muhd about 12 yearsIt would be more useful if you could apply it to background images.
-
Muhd about 12 yearsAnd I think my updated version of Jacob's bookmarklet is way better than the one you linked (especially since it preserves color variety).
-
musefan about 11 years@JacobRelkin: I have found a very simple fix for your code. Just loop the elements in reverse order:
$($("*").get().reverse()).each(...
-
JesseBuesking about 11 yearsHere's a version with @musefan's fix, stripped down and inlined rgbcolor.js, and minified: gist.github.com/JesseBuesking/5260743
-
leosok about 11 yearsYou can also use "-webkit:invert(100%);" I made this Bookmarklet, which will also invert the Page back: jsfiddle.net/nikita_turing/jVKw6/2
-
Muhd about 11 years@leosok That's really awesome... It does images and form elements and everything.
-
leosok about 11 years@Muhd I know. I only now realize that that's special :-) If You think it's worth an answer, I will answer it here!
-
Muhd about 11 years@leosok, I think you should. It will give your script more visibility.
-
renanleandrof over 10 yearsI found a solution for Firefox. I added it to your code. Fiddle here: jsfiddle.net/jVKw6/4
-
renanleandrof over 10 yearsAnd here is a full code that works on both IE and Firefox too! jsfiddle.net/jVKw6/8
-
Oxwivi over 10 years@Renanlf I created a bookmarklet with your JSFiddle code, but nothing happens. Please help.
-
Martin Braun over 9 yearsFor chrome there is an extension doing that job, already: chrome.google.com/webstore/detail/night-reading-mode/…
-
astryk over 8 yearsA minor tweak on the above.
-
Collin Chaffin about 7 yearsThe only answer meeting the OPs requirements of excluding images (and actually works) - is the answer utilizing the RGBColor class (hence why it has today triple the upvotes of the accepted answer). A picture is worth a thousand words: We'll just use the LifeHacker front page as a test.
Original page:
i.imgur.com/1oxgM1d.pngUsing the webkit invert filter via bookmarklet:
i.imgur.com/1RLBjvw.pngUsing the RGBColor class provided above by Jason via bookmarklet (minified version):
i.imgur.com/ghCCVBU.png Any questions?I know which I want to read!:) -
Collin Chaffin about 7 yearsSee my comment above I posted in the accepted answer. Your solution along with the accepted answer is just incorrect does not meet the OPs requirement of EXCLUDING images, and I even posted screenshots using your posted bookmarklet as the invert filter one doing a full close and reload in between each one using the front page of the LifeHacker site as example as you can see, only the solution here that produces a page that is even remotely readable, and actually looks pretty darn good, is the one utilizing Stoyan's RGBcolor class.
-
Collin Chaffin about 7 yearsFirst of all you are absolutely incorrect that the accepted answer is correct I even posted screenshots in the above comments showing just how incorrect it really is. This updated one posted here unfortunately produces exactly the same totally illegible, horrible results that simply invert it all images included, but miss some and are generally not usable, whereas years later Stoyan's RGBcolor class still produces very acceptable results, and also still meets the OPs original question requirement of EXCLUDING images - even after several years of changes to standards, etc.!
-
Jenna Sloan over 6 yearsCouldn't you just do
document.body.parentElement.style='-webkit-filter: invert(100%);-moz-filter: invert(100%);-o-filter: invert(100%);-ms-filter: invert(100%);'
? -
T.Todua over 6 yearsThat code doesnt always work. For example, try to run that code (with Inspect>console) on this page : protectpages.com/blog/night-mode-for-windows-pc
-
ggorlen over 3 yearsYou can reduce the x-ray effect with
hue-rotate
, e.g.document.body.style.cssText = "filter: invert(100%) hue-rotate(180deg) !important;"
.