Solution 1 :

I’ll leave this here for you to have a play with. The last thing I do is in fact putImageData. I guess the problem lies in one of the bits I don’t recognize, as mentioned in the comments.

"use strict";
//window.addEventListener('load', onLoadedCreateB64url, false);
window.addEventListener('load', onLoadedUseEmbeddedImage, false);

// 33x33 pixel image of front wheel
var b64ImgSrc2 = '';

function onLoadedUseEmbeddedImage(evt)
{
    let img = document.querySelector('img');
    img.addEventListener('load', imgLoaded, false);
    img.src = b64ImgSrc2;
}

function imgLoaded(evt)
{
    let img = this;
    let can = document.querySelector('canvas');
    let ctx = can.getContext('2d');
    can.width = img.naturalWidth;
    can.height = img.naturalHeight;
    ctx.drawImage(img, 0, 0);
    
    
    //141,127,143 - a few vertical pixels on right hand side of rim
    
   // color: #8d7f8f;

    recolourInPlace(can, 141,127,143, 255,0,0)
}


function onLoadedCreateB64url(evt)
{
    let img = document.querySelector('img');
    
    let can = document.querySelector('canvas');
    can.width = img.naturalWidth;
    can.height = img.naturalHeight;
    
    let ctx = can.getContext('2d');
    ctx.drawImage(img,0,0);
    
    let src = can.toDataURL();
    console.log(src);
}

function recolourInPlace(canvas, rin,bin,gin, rout,gout,bout)
{
    let ctx = canvas.getContext('2d');
    let imgData = ctx.getImageData(0,0, canvas.width, canvas.height);
    let width = canvas.width, height = canvas.height;
    let data = imgData.data;
    
    for (var y=0; y<height; y++)
    {
        for (var x=0; x<width; x++)
        {
            let index = ((y*width)+x) *4;
            if (data[index+0]==rin && data[index+1]==bin && data[index+2]==gin)
            {
                data[index+0] = rout;
                data[index+1] = gout;
                data[index+2] = bout;
            }
        }
    }
    ctx.putImageData(imgData, 0, 0);
}
    <img/>
    <canvas></canvas>

Problem :

Well I have been trying to make pixel manipulation work with image on canvas.enter image description here

Basically what I am trying to achieve is swapping/replacing of colors present in the image which already been uploaded on the canvas with other custom colors. As you can see in the attached image is my canvas designer with uploaded image and to the left bottom of designer there are colors with bullet points which comes from the image(colors with bullet points are present in the image) and above that is my custom color palette.

Here is the steps I follow for swapping/replacing the colors from the image after image has been uploaded on the canvas

  1. Upload the image to the canvas
  2. Click the check box for the color to swap which comes from an uploaded image on canvas(eg purple checked with checkbox shown in image)
  3. click on any color from custom color palette(eg yellow shown in image) to replace the color in the image(eg purple should get replaced with yellow in the image)

Here is what I have done so far for pixel manipulation.The below part gets triggered as soon as I click on my custom colors after checking box of colors from image

 jQuery('ul').on('click','li.licolors',function() {
                    console.log("licolors");
                    var activeObject = canvas.getActiveObject();
                    if (activeObject && (activeObject.type === 'path')) {
                        activeObject.set("stroke",jQuery(this).css('background-color'));
                        activeObject.dirty = true;
                        canvas.renderAll();
                        updateCanvasState();
                    }
                    else if(activeObject && (activeObject.type === "image")){
                        if($('.checkboxlicolorsfromimage').is(':checked')){
                            var colorfromimage = $('.checkboxlicolorsfromimage').val().split(",");//splitting color of checked checkbox purple rgb(83,70,128,255)
                            var Rcfi = colorfromimage[0];//Red from image
                            console.log(Rcfi);
                            var Gcfi = colorfromimage[1];//Green from image
                            console.log(Gcfi)
                            var Bcfi = colorfromimage[2];//Blue from image
                            console.log("bcfi"+Bcfi);
                            var Acfi = colorfromimage[3];//Alpha from image
                            var colortoimage = getRGB(jQuery(this).css('background-color'));//this variable holds rgbvalue (yellow color rgb(255, 205, 0)) which I have to apply to the image
                            var Rcti = colortoimage.red;//Red apply to image
                            console.log(Rcti);
                            var Gcti = colortoimage.green;//Green apply to image
                            console.log(Gcti);
                            var Bcti = colortoimage.blue;//Blue apply to image
                            console.log(Bcti);
                            var target = new Image();
                            target.src = canvas.toDataURL();
                            canvas.getContext('2d').drawImage(target,0,0);
                            const imageData = canvas.getContext('2d').getImageData(0, 0, target.width, target.height);
                            for (var i = 0; i < imageData.data.length; i += 4) {
                                if(parseInt(imageData.data[i]) == parseInt(Rcti)){console.log(imageData.data[i]);
                                    imageData.data[i] = Rcfi;//Red
                                    console.log(Rcfi);
                                }
                                if(parseInt(imageData.data[i + 1]) == parseInt(Gcti)){console.log(imageData.data[i + 1]);
                                    imageData.data[i + 1] = Gcfi;//Green
                                    console.log(Gcfi);
                                }
                                if(parseInt(imageData.data[i + 2]) == parseInt(Bcti)){console.log(imageData.data[i + 2]);
                                    imageData.data[i + 2] = Bcfi;//Blue
                                    console.log(Bcfi);
                                }
                            }
                            canvas.getContext('2d').putImageData(imageData, 0, 0);
                            activeObject.dirty = true;
                            canvas.renderAll();
                            updateCanvasState();
                        }
                    }
                });

I not getting any errors or warnings I can see in the console that I am able to run the above code but it is not swapping my colors in the already uploaded image on the canvas.

  1. Is it even possible to swap/replace colors in the image after it has been uploaded to the canvas? then swap/replace specific color channels with our own color
  2. Most of the references I found said to be on image .onload
    https://pictureelement.github.io/html5tech/canvas-pixel-manipulation-and-animations.html
    https://developer.mozilla.org/en-US/docs/Web/API/Canvas_API/Tutorial/Pixel_manipulation_with_canvas
    https://www.tutorialspoint.com/Change-colour-of-an-image-drawn-on-an-HTML5-canvas-element

Comments

Comment posted by Thomas Sablik

Have you tried to debug the code with the debugger?

Comment posted by A Haworth

Are you wanting to do exact color matching – because it looks as though your image is quite complex color-wise.

Comment posted by J. Doe

Yes debugged it by putting breakpoints in my browser console and I can see my console.logs prints the pixels in console

Comment posted by enhzflep

My couple of bucks can’t help but eye-off the possibility that you’ve updated your canvas properly, but somehow nuked it again with the

Comment posted by J. Doe

Yes I believe it should be possible to replace the specific color pixels with my custom color pixels not sure if that only works on while image is

By