JavaScript: Invert color on all elements of a page

54,374

Solution 1

My solution seems to work only for Chrome right now, but it inverts everything (including images and iframes) as seen here:

enter image description 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:

alt text

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.

Share:
54,374

Related videos on Youtube

Muhd
Author by

Muhd

SOreadytohelp

Updated on December 24, 2020

Comments

  • Muhd
    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
      Muhd about 12 years
      Apparently you can't inline comments in a bookmarklet since it gets converted into one line.
    • Muhd
      Muhd about 12 years
      I 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
      ggorlen over 3 years
      It'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
    Muhd over 13 years
    +1 This is very nice, but I am having trouble making it into a bookmarklet
  • Jacob Relkin
    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
    Muhd over 13 years
    This 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
    Kranu over 13 years
    +1 I was just browsing, but this bookmarklet will come in handy :)
  • Jacob Relkin
    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
    Muhd over 13 years
    You 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
    Admin over 13 years
    I thing it's StackOverflow Nightshow. well don't show this to jeffatwood. they thing that you make a another clone OF SO.
  • Muhd
    Muhd over 13 years
    A 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
    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
    Muhd over 13 years
    @Jacob, I modified the code to handle iframe contents: jsfiddle.net/F7HqS/5 .
  • Jacob Relkin
    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
    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
    Jacob Relkin over 13 years
    @Muhd Thanks for that - I will take that into consideration.
  • Jacob Relkin
    Jacob Relkin over 13 years
    @Muhd See my latest question. stackoverflow.com/questions/5000108/…
  • Muhd
    Muhd over 13 years
    One 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
    Jacob Relkin about 13 years
    @Muhd Okay thanks. Will let you know when I get around to finishing this.
  • Muhd
    Muhd about 12 years
    It would be more useful if you could apply it to background images.
  • Muhd
    Muhd about 12 years
    And I think my updated version of Jacob's bookmarklet is way better than the one you linked (especially since it preserves color variety).
  • musefan
    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
    JesseBuesking about 11 years
    Here's a version with @musefan's fix, stripped down and inlined rgbcolor.js, and minified: gist.github.com/JesseBuesking/5260743
  • leosok
    leosok about 11 years
    You can also use "-webkit:invert(100%);" I made this Bookmarklet, which will also invert the Page back: jsfiddle.net/nikita_turing/jVKw6/2
  • Muhd
    Muhd about 11 years
    @leosok That's really awesome... It does images and form elements and everything.
  • leosok
    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
    Muhd about 11 years
    @leosok, I think you should. It will give your script more visibility.
  • renanleandrof
    renanleandrof over 10 years
    I found a solution for Firefox. I added it to your code. Fiddle here: jsfiddle.net/jVKw6/4
  • renanleandrof
    renanleandrof over 10 years
    And here is a full code that works on both IE and Firefox too! jsfiddle.net/jVKw6/8
  • Oxwivi
    Oxwivi over 10 years
    @Renanlf I created a bookmarklet with your JSFiddle code, but nothing happens. Please help.
  • Martin Braun
    Martin Braun over 9 years
    For chrome there is an extension doing that job, already: chrome.google.com/webstore/detail/night-reading-mode/…
  • astryk
    astryk over 8 years
    A minor tweak on the above.
  • Collin Chaffin
    Collin Chaffin about 7 years
    The 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.png Using the webkit invert filter via bookmarklet: i.imgur.com/1RLBjvw.png Using 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
    Collin Chaffin about 7 years
    See 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
    Collin Chaffin about 7 years
    First 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
    Jenna Sloan over 6 years
    Couldn'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
    T.Todua over 6 years
    That 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
    ggorlen over 3 years
    You can reduce the x-ray effect with hue-rotate, e.g. document.body.style.cssText = "filter: invert(100%) hue-rotate(180deg) !important;".