10

Edit Png Image For E.g change particular area and fill color in that area.

I want to change the color of an image only selected area. Like the first user selects a color then he chooses an image after that he can change the particular area of image color with the selected color.

After that he can save that image too Is it possible?? like Color company website uses this kind of thing.

like below URL https://www.sherwin-williams.com/visualizer#/active

Partha
  • 402
  • 4
  • 15
user3198563
  • 445
  • 3
  • 13

4 Answers4

9

Edit: Its Possible to get the image data from a canvas and change its pixels. You could also search for algorithms for drawing lines, circles and etc. But the ideia is still the same (Get a canvas image, Edit it, then save it).

Since you are using AngularJS, you can create a directive that gets the color as input and plots that color on a specific area of the image. Then you can use the canvas.toDataURL() or canvas2image() method to parse the canvas into an image with the desired format.

Check the MDN docs for this method here https://developer.mozilla.org/pt-BR/docs/Web/API/HTMLCanvasElement/toDataURL

I have made a small code snippet just to give you an ideia. I think that the color picker feature is a step that you can easely implement and deliver as input to your canvas directive. I hope that helps!

function hexToRgb(color) {
    var shorthandRegex = /^#?([a-f\d])([a-f\d])([a-f\d])$/i;
    color = color.replace(shorthandRegex, function(m, r, g, b) {
        return r + r + g + g + b + b;
    });
    
    var result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(color);
    return result ? {
        r: parseInt(result[1], 16),
        g: parseInt(result[2], 16),
        b: parseInt(result[3], 16)
    } : {
        r: 0,
        g: 0,
        b: 0
    };
}

function colorImage(imgId,hexaColor) {
    // create hidden canvas (using image dimensions)
    var imgElement = document.getElementById(imgId);
  
    var canvas = document.createElement("canvas");
    canvas.width = imgElement.width;
    canvas.height = imgElement.height;

    var ctx = canvas.getContext("2d");
    ctx.drawImage(imgElement,0,0);
    
    var imageData = ctx.getImageData(0,0,canvas.width,canvas.height);
    
    var data = imageData.data;
  
    // convert image to grayscale
    var rgbColor = hexToRgb(hexaColor);
    //console.log(rgbColor);
    
    for(var p = 0, len = data.length; p < len; p+=4) {
        //if(data[p+3] == 0)
        //   continue;
        data[p + 0] = rgbColor.r;
        data[p + 1] = rgbColor.g;
        data[p + 2] = rgbColor.b;
        //data[p + 3] = 255;
    }
    ctx.putImageData(imageData, 0, 0);

    // replace image source with canvas data
    imgElement.src = canvas.toDataURL();
}

// changing color of capsule on select event
document.getElementById('sel_top').onchange = function(){
    colorImage('img_top', this.value);
}
document.getElementById('sel_bottom').onchange = function(){
    colorImage('img_bottom', this.value);
}

$('input[type="color"]').on("change", function(color) {
      console.log(this.value);
      var colorHex = this.value;
   
      colorImage('img_top',colorHex);
 });
img.color-img {
 position: absolute;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.3.1/jquery.slim.js"></script>

<script src="https://cdnjs.cloudflare.com/ajax/libs/tinyColorPicker/1.1.1/jqColorPicker.min.js"></script>

<label for="sel_top">top color: </label>
  <select id="sel_top">
   <option value="#ffffff">select</option>
      <option value="#ff0000">red</option>
      <option value="#00ff00">green</option>
      <option value="#0000ff">blue</option>
      <option value="#FCF153">yellow</option>    
  </select>
  <label for="sel_top">bottom color: </label>
  <select id="sel_bottom">
      <option value="#ffffff">select</option>
      <option value="#ff0000">red</option>
      <option value="#00ff00">green</option>
      <option value="#0000ff">blue</option>
      <option value="#FCF153">yellow</option>    
  </select>
    <div>
      <input type="color" value="#ff0000" class="myColorPic">
    </div>
  <br><br>
  <img src="" />
  
  <img id="img_top" src="" class="color-img" />
  <img id="img_bottom" src="" class="color-img" />
  <img src="" class="color-img" />
Albert Einstein
  • 7,472
  • 8
  • 36
  • 71
Tiago Conceiçao
  • 457
  • 1
  • 5
  • 17
  • 1
    Hi i want to change color of an image , daont want to draw – user3198563 Jun 12 '18 at 04:32
  • Still the ideia is the same. I have made some changes on the code in order to change the image color. I hope it helps! – Tiago Conceiçao Jun 12 '18 at 12:14
  • thanks for your reply... can you make like first user picks a color then he can select any area of an image and selected color will go there. Area means a home inner image – user3198563 Jun 13 '18 at 03:24
  • To select an area and do whatever you want to the selected image please check similar questions like here https://stackoverflow.com/questions/35316667/how-to-make-selection-box-in-canvas-to-select-any-object-it-touches-not-only-obj and here https://stackoverflow.com/questions/32752547/html5-canvas-select-an-area-and-zoom – Tiago Conceiçao Jun 13 '18 at 11:34
  • you referred two urls . both are not usable for me. Please check this url please. – user3198563 Jun 14 '18 at 05:55
  • https://www.sherwin-williams.com/visualizer#/active please see this url – user3198563 Jun 14 '18 at 05:58
  • can you show example where user can choose color not from drop down but from color picker after that he can select particular area of an image add selected color will fill on that area – user3198563 Jun 19 '18 at 03:52
  • @user3198563 I have Added a color Pic to the snnipet. For image areas selection he could check a Jquery plugin that gives the coordinates of the selected area. I found a plugin on this link http://odyniec.net/projects/imgareaselect/ – Tiago Conceiçao Jun 19 '18 at 12:04
  • hii that is not working ..... see i have two walls . i want to fill first wall only with selected color .. please helped me out – user3198563 Jun 20 '18 at 10:40
4

Demo:

Full page: https://angularjs-rj88cu.stackblitz.io/

Code: https://stackblitz.com/edit/angularjs-rj88cu

Explanation:

The editor is an svg containing background image as <image> and areas as <path> both are created from editor.imageUrl and editor.areas

For downloading,

  1. Background image is drawn on a canvas
  2. SVG (without <image>) is converted to dataUrl and then drawn on canvas
  3. Canvas is converted to dataUrl for downloading

Why canvas was not used instead of svg in the first place?

Because mouse interaction are difficult to implement in canvas and is much easier in inline svg as the work like DOM elements (hover pseudo class, click events, etc)

Also I assumed you wanted angularjs because you have tagged it (even though you have not mentioned in the question)

Also there are a lot of bad practices in the code like editor not being a component and styling input[type=color]'s Shadow DOM (you can use some colorpicker plugin instead)

PS: Tell me if something needs to be changed

Devansh J
  • 4,006
  • 11
  • 23
  • hii thanks for your reply . Can you describe full code here and it will be very helpful if you see sherwin-williams.com/visualizer#/active this url. Need something like this – user3198563 Jun 25 '18 at 03:59
  • @user3198563 1. The code is pretty self-explanatory, you can ask me which parts you didn't understand. 2. I have made as per the link in the first place and it is very similar, if you want something additional please be specific and I'll help you out. 3. Do you think my answer is the best at this point as compared to other answers? – Devansh J Jun 25 '18 at 09:38
  • yes as compared to other your answer is good. See first i have to select color from color picker, in second step i have to select any area of the pic . Then selected area will be automatically filed with selected color – user3198563 Jun 26 '18 at 05:29
  • and can you use full image of wall – user3198563 Jun 26 '18 at 05:32
  • I'll update the color change mechanism soon... And what do mean by "full image of wall"? – Devansh J Jun 26 '18 at 08:16
  • Let us [continue this discussion in chat](https://chat.stackoverflow.com/rooms/173801/discussion-between-user3198563-and-devansh-j). – user3198563 Jun 26 '18 at 08:47
0

The functional in the example - custom product, and it's implemented through AngularJS.

If you want to do something like this, you may combine several jQuery plugins and HTML5 features.

Things you need:

  • some jQuery color picker - for choose color
  • HTML5 canvas - for show borders and fill image areas with a color
  • canvas2image - to save image

If you want to use some out of the box, just look for some jquery colorize image plugin.

zen
  • 980
  • 6
  • 18
-1

Just copy and paste this function in custom helper file or your class. You just need to set parameter for proper image resolution result.

  function resize($source_image, $destination, $tn_w, $tn_h, $quality = 100, $wmsource = false) {
        $info = getimagesize($source_image);
        $imgtype = image_type_to_mime_type($info[2]);

        #assuming the mime type is correct
        switch ($imgtype) {
            case 'image/jpeg':
            $source = imagecreatefromjpeg($source_image);
            break;
            case 'image/jpg':
            $source = imagecreatefromjpeg($source_image);
            break;
            case 'image/gif':
            $source = imagecreatefromgif($source_image);
            break;
            case 'image/png':
            $source = imagecreatefrompng($source_image);
            break;
            default:
            die('Invalid image type.');
        }
        #Figure out the dimensions of the image and the dimensions of the desired thumbnail
        $src_w = imagesx($source);
        $src_h = imagesy($source);


        #Do some math to figure out which way we'll need to crop the image
        #to get it proportional to the new size, then crop or adjust as needed

        $x_ratio = $tn_w / $src_w;
        $y_ratio = $tn_h / $src_h;

        if (($src_w <= $tn_w) && ($src_h <= $tn_h)) {
            $new_w = $src_w;
            $new_h = $src_h;         
        } elseif (($x_ratio * $src_h) < $tn_h) {
            $new_h = ceil($x_ratio * $src_h);
            $new_w = $tn_w;
        } else {
            $new_w = ceil($y_ratio * $src_w);
            $new_h = $tn_h;
        }

        $newpic = imagecreatetruecolor(round($new_w), round($new_h));
        imagecopyresampled($newpic, $source, 0, 0, 0, 0, $new_w, $new_h, $src_w, $src_h);
        $final = imagecreatetruecolor($tn_w, $tn_h);
        $backgroundColor = imagecolorallocate($final, 248,248,248);
        imagefill($final, 0, 0, $backgroundColor);
        //imagecopyresampled($final, $newpic, 0, 0, ($x_mid - ($tn_w / 2)), ($y_mid - ($tn_h / 2)), $tn_w, $tn_h, $tn_w, $tn_h);
        imagecopy($final, $newpic, (($tn_w - $new_w)/ 2), (($tn_h - $new_h) / 2), 0, 0, $new_w, $new_h);
        

        #if we need to add a watermark
        if ($wmsource) {
            #find out what type of image the watermark is
            $info    = getimagesize($wmsource);
            $imgtype = image_type_to_mime_type($info[2]);

            #assuming the mime type is correct
            switch ($imgtype) {
                case 'image/jpeg':
                $watermark = imagecreatefromjpeg($wmsource);
                break;
                case 'image/gif':
                $watermark = imagecreatefromgif($wmsource);
                break;
                case 'image/png':
                $watermark = imagecreatefrompng($wmsource);
                break;
                default:
                die('Invalid watermark type.');
            }

            #if we're adding a watermark, figure out the size of the watermark
            #and then place the watermark image on the bottom right of the image
            $wm_w = imagesx($watermark);
            $wm_h = imagesy($watermark);
            imagecopy($final, $watermark, $tn_w - $wm_w, $tn_h - $wm_h, 0, 0, $tn_w, $tn_h);

        }
        if (imagejpeg($final, $destination, $quality)) {
            return true;
        }
        return false;
    }
Davesexcel
  • 6,896
  • 2
  • 27
  • 42