How to make the "cursor:" defined cursor-image renders as expected in browser on a retina screen?

364 views Asked by At

Try to copy this CSS definition in body on this page:

div{

    cursor: url(""), auto !important;
    height:200px;
    width:200px;
}
<div>hover</div>

The customized cursor works but it's not rendered as a retina image. It's rendered as a blurry, 100% bigger-sized image.

Is there any way to force it to display as a retina image? (I tried both Chrome and Safari)

2

There are 2 answers

0
awholegnuworld On

Here is some CSS. I just converted your PNG to a SVG and resized it. Feel free to decode this and resize the SVG in the XML.

div{

    cursor: url(""), auto;
    height:200px;
    width:200px;
}
0
Brandon McConnell On

Yes, this is fully possible using SVG, which displays in crisp definition on screens of any pixel density, standard (@1x), retina (@2x), and even 4k and 8k displays (@3x, @4x+).

You also don't have to use Base64, and it may be easier without relying on Base64. Here is an example without Base^4, where I simply pull the cursor SVG image in via URL. I also have a dynamic Base64 example below this first example, so scroll down for that.

div {
  cursor: url('https://assets.codepen.io/1580009/stackoverflow-67356313-cursor.svg'), auto;
  height: 200px;
  width: 200px;
}
<div>hover</div>

If you would prefer to Base64 encode your SVG, that will work too, though it's not always performant (Probably Don't Base64 SVG). Below is an example which uses Base64 dynamically. For the sake of this example, I've added the SVG itself into the body of the HTML, along with a field that you can use to control the size of the cursor. As you adjust the input field's cursor size, the Base64 string will automatically recompute the new Base64 string for the updated cusror size and rewrite that style on the hover-div.

const input = document.querySelector('input'),
      svgCursor = document.querySelector('svg#cursor'),
      hoverDiv = document.querySelector('div');

const setCursor = (size = 16) => {
  svgCursor.setAttribute('width', size);
  svgCursor.setAttribute('height', size);
  const base64cursor = window.btoa(new XMLSerializer().serializeToString(svgCursor));
  hoverDiv.style.setProperty('cursor', `url('data:image/svg+xml;base64,${base64cursor}'), auto`);
};
setCursor();

input.addEventListener('change', () => setCursor(input.value));
div {
  margin-top: 20px;
  height: 200px;
  width: 200px;
  border: 2px dashed #f00;
}
svg#cursor {
  display: none;
}
<label>Cursor size: <input value="16" type="number"></label>

<div>hover</div>

<svg version="1.1" id="cursor" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" viewBox="0 0 32 32" style="enable-background:new 0 0 32 32;" xml:space="preserve" width="16" height="16"><polygon fill="#D0CBE7" points="0,0 13,32 17,23 25,32 32,25 23.7,17.1 32,13 "/><g><polygon fill="#EDE9FA" points="24,17 32,13 0,0 29,28 32,25"/></g></svg>