DataURL of an img with CSS class


DataURL of an img with CSS class



I have to apply some styles on <img> thanks to a CSS class.


<img>



Is it possible to get the dataURL of the <img> with the CSS style ?


dataURL


<img>




$(function() {
// Original
const imgOriginal = document.getElementById('original');
const c1 = document.getElementById('c1');
let ctx = c1.getContext('2d');
ctx.drawImage(imgOriginal, 100, 100);

// Filtered
const imgFiltered = document.getElementById('filtered');
const c2 = document.getElementById('c2');
ctx = c2.getContext('2d');
ctx.drawImage(imgFiltered, 100, 100);

// Same dataURL :(
console.log(c1.toDataURL(), c2.toDataURL());
console.log(c1.toDataURL() === c2.toDataURL());
})


.filter::before {
display: block;
height: 100%;
left: 0;
position: absolute;
top: 0;
width: 100%;
z-index: 1;
border: 1px solid red;
}

.filter {
position: relative;
-webkit-filter: sepia(.5) hue-rotate(-30deg) saturate(1.4);
filter: sepia(.5) hue-rotate(-30deg) saturate(1.4);
}

canvas {
display: block;
width: 100px;
height: 100px;
border: 1px solid red;
}


https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js












Maybe snippet is going to have a bug because of the <canvas> tag, the idea is there anyway.


<canvas>



EDIT :



If anyone has a suggestion with SVG or something else, I'm using fabricJS.


SVG



EDIT 2 (NOT RESOLVE BUT FIND OTHER WAY) :



Thanks to @KavianK. you could replicate CSS style with the canvas context, however to me it's boring because we have to store a different callback for each CSS class to get the dataURL. Working anyway!


CSS


canvas


callback


CSS


dataURL



Thanks to @Emeeus maybe a solution provide from your backend, not solution for me beacause i'm want to do this ONLY on the front-end. wkhtmltopdf



Thanks to @pegasuspect we can filter an image with SVG, I'm following this way and I replace fabricJS by svgjs, this librairie can replace easly a canvas and it's more easier to work with img and I dind't need the DataURL anymore !


SVG


canvas


img


DataURL



Thanks to @Kaiido there is a way to take a snapshot of your HTML rendered with CSS style with html2canvas easy to get dataURL with this case. Unfortunataly some CSS styles are not supported yet like box-shadow or filter that's why it's not a solution for me


snapshot


HTML


CSS


dataURL


CSS


box-shadow


filter



This topic is not resolve but with svgjs I don't need actually work with dataURL.


svgjs


dataURL





I think it can't. since you can manipulate the pixels in the canvas, you can implement how you edit the image.
– kiro
Jun 21 at 15:31





Just to understand, you want to get rid of javascript ? Or you just want the filtering to be done by css classes ? What should be modified in @kavian K. answer in order to be ok for you ?
– Mium
Jun 28 at 9:09





@Mium indeed, I want to know if is it possible to do this without use the canvas's context. I'm trying with SVG at the moment maybe it's a good way : developer.mozilla.org/en-US/docs/Web/SVG/…
– GameTag
Jun 28 at 9:26



canvas's context


SVG





Well unless your images are svg formatted, I don't think you can export the image+css into a dataUrl without going through canvas. Simply because the bitmap (what is displayed in your screen) is a rendered version, and what you can access through javascript is "raw" data, not combined nor rendered. So the best way today is to use canvas api to capture the bitmap (the rendered version). @Kavian K. answer seems a good solution.
– Mium
Jun 28 at 9:42





Possible duplicate of stackoverflow.com/questions/49291239/…
– Kaiido
2 days ago




5 Answers
5



CSS and DOM is a separate world from the bitmaps that are used for images and canvas. The bitmaps themselves are not affected by CSS, only the elements which acts as a looking-glass to the bitmap. So, CSS filters applied to the canvas will not be applied to the image that is produced. You either need to replicate the filters in canvas or rather re apply the same filters to the generated image.


CSS


DOM


CSS



Example:



There is a little known property on the context object, conveniently named filter. This will apply a filter on the context it self. The filter must be set before next draw operation.




var img = new Image();

img.crossOrigin = '';
img.src = document.getElementById( 'original' ).src;

img.onload = function() {
var canvas = document.getElementById( 'canvas' ),
ctx = canvas.getContext( '2d' );

canvas.width = this.width;
canvas.height = this.height;

// filter
if ( typeof ctx.filter !== 'undefined' ) {
ctx.filter = "sepia(.5) hue-rotate(-30deg) saturate(1.4)";
ctx.drawImage(this, 0, 0);
} else {
ctx.drawImage(this, 0, 0);
}

document.getElementById( 'filtered' ).src = canvas.toDataURL();
}


<img id="original" src="https://upload.wikimedia.org/wikipedia/commons/thumb/c/c7/Brad_Pitt_Inglorious_Basterds_Berlin_premiere.jpg/170px-Brad_Pitt_Inglorious_Basterds_Berlin_premiere.jpg" />
<img id="filtered" />
<canvas id="canvas" style="display: none"></canvas>





Hey, I know about filter() function but I wanna to know if there is other way to do this without have to store all effects in a Javascript object. That's why I want kind of snapshot of <img> + CSS => toDataUrl(). Thanks anyway
– GameTag
Jun 21 at 17:22



filter()


<img> + CSS


toDataUrl()



tldr;



You can do it with SVG. http://jsfiddle.net/1hambw93/91/



It is explained here quite nicely. It basically says you can use svg element in src tag of an img.


img



It is explained here quite nicely as well: I could achieve the same filter effect with your code, using an SVG filter.



You can generate filter for svg from this site using Sepium to get the same filter as your css. You would have the following SVG so far:


<svg id="test">
<image xlink:href='https://upload.wikimedia.org/wikipedia/commons/thumb/c/c7/Brad_Pitt_Inglorious_Basterds_Berlin_premiere.jpg/170px-Brad_Pitt_Inglorious_Basterds_Berlin_premiere.jpg'
x="0" y="0" height="170px" width="218px"
filter='url(#sepium-filter)' />
<filter id="sepium-filter">
<feColorMatrix type="matrix"
values="1.3 -0.3 1.1 0 0
0 1.3 0.2 0 0
0 0 0.8 0.2 0
0 0 0 1 0">
</feColorMatrix>
</filter>
</svg>



Then I used a minimal javascript code to convert to base64 and write the result from the SVG to an object, to display in HTML.





Thanks to this, I never manipulate SVG before, it's ***** powerfull, I replaced fabricJS with svg.js I'll can do what I want without a canvas
– GameTag
yesterday


SVG


fabricJS


svg.js


canvas





Nice solution @pegasuspect
– Emeeus
yesterday





@Emeeus thank you. GameTag yes it is. I didn’t know that either. It is you who gave me the clue but I was not sure if that was even possible! Cheers to all 🍻👍🏼
– pegasuspect
yesterday



As I understand, You need the computed image, so you need the result of the work made by the browser. To accomplish that, you could use wkhtmltopdf specifically
wkhtmltoimage. It uses Qt WebKit rendering engine (just like a browser). You have to install that in the server and run somethig like:


wkhtmltoimage http://mysite/image-Plus-css-PLus-canvas.html myComputedImge.jpg



Where http://mysite/image-Plus-css-PLus-canvas.html is your image in the html with the css, javascript and whatever. All of this of course could be accomplished using ajax.



So, using this way you have a .jpg or .png file (myComputedImge.jpg in this case) that has all what you want computed, like a screenshot. If you want the base64 you could do the same as you do with .toDataURL() and the result is the base64 of the image with the css.


.toDataURL()



If you want to perform this using ajax, you could:


.toDataURL()





I don't want use the backend to do this. I'll precise this on the main post. Thanks anyway to share this !
– GameTag
2 days ago





@GameTag maybe the explanation sounds a bit confusing but all of this could be resolved with a few lines of code.
– Emeeus
2 days ago




There is 1 work around; It might work for your use case using CSS:


img[src^="data:image/png;"] {
background-color: #000;
filter: sepia(50%);
}



This will work for all the images URLs starting with );
background-repeat:no-repeat;
height:240px;
width:auto;
}





To get the data url I downloaded the image and then used a small javascript function to retrieve the information from the uploaded information. The function could be tweaked to use online sources but this one was saved so... ("here's one i made earlier!")..



Function for future reference: (should link fail)




function encodeImageFileAsURL(cb) {
return function(){
var file = this.files[0];
var reader = new FileReader();
reader.onloadend = function () {
cb(reader.result);
}
reader.readAsDataURL(file);
}
}

$('#inputFileToLoad').change(encodeImageFileAsURL(function(base64Img){
$('.output')
.find('textarea')
.val(base64Img)
.end()
.find('a')
.attr('href', base64Img)
.text(base64Img)
.end()
.find('img')
.attr('src', base64Img);
}));


body{
padding: 20px;
}


https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js
<link href="//netdna.bootstrapcdn.com/bootstrap/3.1.0/css/bootstrap.min.css" rel="stylesheet"/>
<h2>Input</h2>
<form class="input-group" id="img2b64">
<input id="inputFileToLoad" type="file" onchange="encodeImageFileAsURL();" />
</form>

<hr>

<h2>Output</h2>










you show me how to display an image with DataURL but it's not answer to my question. Is it possible to get DataURL of <img class="filter"> with her CSS style. Thanks anyway !
– GameTag
Jun 25 at 7:05


DataURL


DataURL


<img class="filter">


CSS





Apply the filter, download the image and get the data URL using the function. If it's different from the original, well then it's possible!
– Rachel Gallen
Jun 25 at 9:15





I edit my post, maybe solution with SVG, I don't understand why I should download image, can you detail ? Thanks !
– GameTag
Jun 26 at 19:00


SVG





I can see your comment in my inbox but not on this post, so maybe you deleted it? The reason I said to download the image was so you could use the fiddle provided to upload 5he image and test whether the data URL s were the same. If they are same, then the filters don't affect the data URL, if they are different, well then it's possible to get the data URL of an image with filters applied, and you can manipulate the function and apply it to your code.
– Rachel Gallen
Jun 26 at 19:08






By clicking "Post Your Answer", you acknowledge that you have read our updated terms of service, privacy policy and cookie policy, and that your continued use of the website is subject to these policies.

Comments

Popular posts from this blog

paramiko-expect timeout is happening after executing the command

Export result set on Dbeaver to CSV

Opening a url is failing in Swift