Circular images usually have this problem when compressed, mainly because there is no "half-pixel". Look at the images below.
data:image/s3,"s3://crabby-images/1e950/1e950f178ad11db9c899e388580d8e08ce7ca4fb" alt=""
Imagesinhighpixeldensitypanels,suchasApple'sRetinapanel
data:image/s3,"s3://crabby-images/8b7c4/8b7c4fbcc237e95497cd200609789db23b0a2820" alt=""
BrowsersdonotrenderimagesaswellasPhotoshopforexample.SosometimesinPhotoshopitlooksgood,buttheBrowseralgorithmmaynothavethesameresultcompressingimages.
Mostlikelyifyourimagewasasquarewithlinesleftover,youwouldnothavethisproblemsosharply.
Theidealwouldbetotreattheimageinsomesoftwaremadeforit.Orturninto.SVG.Treatingthiskindofthinginthebrowserformeisgambiarra.Youcanstillusetheimage-rendering:crisp-edges;
classintheimage.
img{image-rendering:crisp-edges;}
Hereyouhavedocumentationabouttheclass: link
In spite of this, you can see how even with it%% of rendering in Browser is different.
Chrome:
data:image/s3,"s3://crabby-images/e74a2/e74a2544fa7710c637479854400055262c8a9364" alt=""
Safari: data:image/s3,"s3://crabby-images/5c3ed/5c3ed62a9cd474e3672d07bf4b28a193a662510c" alt=""
FireFoxQuantum: data:image/s3,"s3://crabby-images/5145c/5145c0ce3f98c85aa351b85b06f49f32e38afd03" alt=""
YoucanalsotreattheimagewithJavaScriptwithinimage-rendering:pixelated;
,asinthisexample:
TotestwhereFalseputTrueinscript
var canvas = document.getElementById('canvas');
var ctx = canvas.getContext('2d');
var textarea = document.getElementById('code');
var reset = document.getElementById('reset');
var edit = document.getElementById('edit');
var code = textarea.value;
function drawCanvas() {
ctx.clearRect(0, 0, canvas.width, canvas.height);
eval(textarea.value);
}
reset.addEventListener('click', function() {
textarea.value = code;
drawCanvas();
});
edit.addEventListener('click', function() {
textarea.focus();
})
textarea.addEventListener('input', drawCanvas);
window.addEventListener('load', drawCanvas);
<canvas id="canvas" width="400" height="200" class="playable-canvas"></canvas>
<div class="playable-buttons">
<input id="edit" type="button" value="Edit" />
<input id="reset" type="button" value="Reset" />
</div>
<textarea id="code" class="playable-code" style="height:140px;">
var img = new Image();
img.src = 'https://mdn.mozillademos.org/files/222/Canvas_createpattern.png';
img.onload = function() {
ctx.mozImageSmoothingEnabled = false;
ctx.webkitImageSmoothingEnabled = false;
ctx.msImageSmoothingEnabled = false;
ctx.imageSmoothingEnabled = false;
ctx.drawImage(img, 0, 0, 400, 200);
};</textarea>