Сейчас 2018 год, и у нас наконец-то есть дешевые способы что-то сделать ...
Действительно, поскольку 2-ый контекстный API теперь имеет свойство filter
, и этоэто свойство фильтра может принимать SVGFilters , мы можем создать SVGFilter, который будет хранить только полностью непрозрачные пиксели в наших чертежах и, таким образом, исключит сглаживание по умолчанию.
, поэтому он не будет деактивированСглаживание само по себе, но обеспечивает дешевый способ как с точки зрения реализации, так и с точки зрения производительности удалить все полупрозрачные пиксели во время рисования.
Я не специалист по SVGFilters, так что может быть лучший способделая это, но для примера я буду использовать узел <feComponentTransfer>
, чтобы захватывать только полностью непрозрачные пиксели.
var ctx = canvas.getContext('2d');
ctx.fillStyle = '#ABEDBE';
ctx.fillRect(0,0,canvas.width,canvas.height);
ctx.fillStyle = 'black';
ctx.font = '14px sans-serif';
ctx.textAlign = 'center';
// first without filter
ctx.fillText('no filter', 60, 20);
drawArc();
drawTriangle();
// then with filter
ctx.setTransform(1, 0, 0, 1, 120, 0);
ctx.filter = 'url(#remove-alpha)';
// and do the same ops
ctx.fillText('no alpha', 60, 20);
drawArc();
drawTriangle();
// to remove the filter
ctx.filter = 'none';
function drawArc() {
ctx.beginPath();
ctx.arc(60, 80, 50, 0, Math.PI * 2);
ctx.stroke();
}
function drawTriangle() {
ctx.beginPath();
ctx.moveTo(60, 150);
ctx.lineTo(110, 230);
ctx.lineTo(10, 230);
ctx.closePath();
ctx.stroke();
}
// unrelated
// simply to show a zoomed-in version
var zCtx = zoomed.getContext('2d');
zCtx.imageSmoothingEnabled = false;
canvas.onmousemove = function drawToZoommed(e) {
var x = e.pageX - this.offsetLeft,
y = e.pageY - this.offsetTop,
w = this.width,
h = this.height;
zCtx.clearRect(0,0,w,h);
zCtx.drawImage(this, x-w/6,y-h/6,w, h, 0,0,w*3, h*3);
}
<svg width="0" height="0" style="position:absolute;z-index:-1;">
<defs>
<filter id="remove-alpha" x="0" y="0" width="100%" height="100%">
<feComponentTransfer>
<feFuncA type="discrete" tableValues="0 1"></feFuncA>
</feComponentTransfer>
</filter>
</defs>
</svg>
<canvas id="canvas" width="250" height="250" ></canvas>
<canvas id="zoomed" width="250" height="250" ></canvas>
А для тех, кому не нравится добавлять элемент <svg>
в их DOM, вы также можете сохранить его как внешний файл SVG и установитьсвойство filter
до path/to/svg_file.svg#remove-alpha
.