javascript - 使用 Javascript 将油画/素描效果应用于照片

标签 javascript html canvas filter photo

我想使用 javascript 从照片开始模拟人体绘画效果。

我一直在搜索几个进行图像处理的 js 库(主要是在 Canvas 上)。 但似乎没有人尝试过我正在寻找的东西。

photoshop oil paint filter

我不认为用 javascript 实现这样的效果是不可能的。所以我想知道为什么我找不到任何已经完成的事情。

在 native 方面,有多种 photoshop 替代方案可以实现此类效果,如 App Store 中的多个应用所示:

以下是可能结果的其他示例(来自 Artist's Touch App):

Artist's Touch App

最佳答案

好吧,我找到了对所用算法的很好解释 here 并适配 JS 和 canvas。

Live Demo

CodePen Demo with controls to mess with the effect

effect result

它的工作原理是对中心像素周围的所有像素进行平均,然后将该平均值乘以所需的强度级别,然后除以 255。然后增加与强度级别相关的 r/g/b .然后检查相邻像素中哪个强度级别最常见,并为目标像素分配该强度级别。

edit 对它做了更多的工作并重写了很多,获得了一些非常巨大的性能提升,现在可以很好地处理大小合适的图像。

var canvas = document.getElementById("canvas"),
    ctx = canvas.getContext("2d"),
    img = new Image();

img.addEventListener('load', function () {
    canvas.width = this.width;
    canvas.height = this.height;
    ctx.drawImage(this, 0, 0, canvas.width, canvas.height);
    oilPaintEffect(canvas, 4, 55);
});

img.crossOrigin = "Anonymous";
img.src = "https://fbcdn-sphotos-h-a.akamaihd.net/hphotos-ak-xpa1/v/t1.0-9/1379992_10202357787410559_1075078295_n.jpg?oh=5b001e9848796dd942f47a0b2f3df6af&oe=542F3FEF&__gda__=1412145968_4dbb7f75b385770ecc3f4b88105cb0f8";

function oilPaintEffect(canvas, radius, intensity) {
    var width = canvas.width,
        height = canvas.height,
        imgData = ctx.getImageData(0, 0, width, height),
        pixData = imgData.data,
        destCanvas = document.createElement("canvas"),
        dCtx = destCanvas.getContext("2d"),
        pixelIntensityCount = [];

    destCanvas.width = width;
    destCanvas.height = height;

    // for demo purposes, remove this to modify the original canvas
    document.body.appendChild(destCanvas);

    var destImageData = dCtx.createImageData(width, height),
        destPixData = destImageData.data,
        intensityLUT = [],
        rgbLUT = [];

    for (var y = 0; y < height; y++) {
        intensityLUT[y] = [];
        rgbLUT[y] = [];
        for (var x = 0; x < width; x++) {
            var idx = (y * width + x) * 4,
                r = pixData[idx],
                g = pixData[idx + 1],
                b = pixData[idx + 2],
                avg = (r + g + b) / 3;

            intensityLUT[y][x] = Math.round((avg * intensity) / 255);
            rgbLUT[y][x] = {
                r: r,
                g: g,
                b: b
            };
        }
    }


    for (y = 0; y < height; y++) {
        for (x = 0; x < width; x++) {

            pixelIntensityCount = [];

            // Find intensities of nearest pixels within radius.
            for (var yy = -radius; yy <= radius; yy++) {
                for (var xx = -radius; xx <= radius; xx++) {
                    if (y + yy > 0 && y + yy < height && x + xx > 0 && x + xx < width) {
                        var intensityVal = intensityLUT[y + yy][x + xx];

                        if (!pixelIntensityCount[intensityVal]) {
                            pixelIntensityCount[intensityVal] = {
                                val: 1,
                                r: rgbLUT[y + yy][x + xx].r,
                                g: rgbLUT[y + yy][x + xx].g,
                                b: rgbLUT[y + yy][x + xx].b
                            }
                        } else {
                            pixelIntensityCount[intensityVal].val++;
                            pixelIntensityCount[intensityVal].r += rgbLUT[y + yy][x + xx].r;
                            pixelIntensityCount[intensityVal].g += rgbLUT[y + yy][x + xx].g;
                            pixelIntensityCount[intensityVal].b += rgbLUT[y + yy][x + xx].b;
                        }
                    }
                }
            }

            pixelIntensityCount.sort(function (a, b) {
                return b.val - a.val;
            });

            var curMax = pixelIntensityCount[0].val,
                dIdx = (y * width + x) * 4;

            destPixData[dIdx] = ~~ (pixelIntensityCount[0].r / curMax);
            destPixData[dIdx + 1] = ~~ (pixelIntensityCount[0].g / curMax);
            destPixData[dIdx + 2] = ~~ (pixelIntensityCount[0].b / curMax);
            destPixData[dIdx + 3] = 255;
        }
    }

    // change this to ctx to instead put the data on the original canvas
    dCtx.putImageData(destImageData, 0, 0);
}

关于javascript - 使用 Javascript 将油画/素描效果应用于照片,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24222556/

相关文章:

c# - 在单选按钮列表中选择选项时更改图像

javascript - css - 如何使 svgs 响应?

PHP 导致服务器错误

css - 如何评价我的 ASP.NET MVC3 HTML5 Web 应用程序的可访问性?

javascript - 将 Canvas 的属性宽度设置为百分比值

javascript - 更简单的 JavaScript 继承,无需重复代码

javascript - 将 y 刻度扩展到 d3 条形图中所需数字的方法

html - CSS,如何创建最长包含文本的标签宽度?

javascript - 如何在 HTML5 中使用动态 x、y 值生成 Canvas 移动波浪?

javascript - 在网格中绘制单元格