Эффект вождения пальцем по воде

120
11 августа 2019, 12:20

Не знал, как точно сформировать вопрос, поэтому сразу прошу прощения.

У меня есть картинка (svg) с линиями, они не прямые, но это не важно. Мне нужен плагин или что то другое, который при неведении курсора будет создавать эффект, как будто водить пальцем по воде. То есть в том месте, где курсор, линии как бы расходятся, а когда убрать курсор - снова стают на место

Answer 1

Вот вариант водной поверхности на canvas, за основу взято демо от Jonas Wagner.

var floor = Math.floor; 
 
function loadImage(callback) { 
    var img = document.getElementById('bg'); 
    callback(img); 
} 
 
function getDataFromImage(img) { 
    canvas.width = img.width; 
    canvas.height = img.height; 
    ctx.clearRect(0, 0, img.width, img.height); 
    ctx.drawImage(img, 0 ,0); 
    return ctx.getImageData(0, 0, img.width, img.height); 
} 
 
function disturb(x, y, z){ 
    if(x < 2 || x > width-2 || y < 1 || y > height-2) 
        return; 
    var i = x+y*width; 
    buffer0[i] += z; 
    buffer0[i-1] -= z; 
} 
 
function process(){ 
    var img = ctx.getImageData(0, 0, width, height), 
        data = img.data, 
        i, x; 
 
    // average cells to make the surface more even 
    for(i=width+1;i<size-width-1;i+=2){ 
        for(x=1;x<width-1;x++,i++){ 
            buffer0[i] = (buffer0[i]+buffer0[i+1]+buffer0[i-1]+buffer0[i-width]+buffer0[i+width])/5; 
        } 
    } 
 
    for(i=width+1;i<size-width-1;i+=2){ 
        for(x=1;x<width-1;x++,i++){ 
            // wave propagation 
            var waveHeight = (buffer0[i-1] + buffer0[i+1] + buffer0[i+width] + buffer0[i-width])/2-buffer1[i]; 
            buffer1[i] = waveHeight; 
            // calculate index in the texture with some fake referaction 
            var ti = i+floor((buffer1[i-2]-waveHeight)*0.08)+floor((buffer1[i-width]-waveHeight)*0.08)*width; 
            // clamping 
            ti = ti < 0 ? 0 : ti > size ? size : ti; 
            // some very fake lighting and caustics based on the wave height 
            // and angle 
            var light = waveHeight*2.0-buffer1[i-2]*0.6, 
                i4 = i*4, 
                ti4 = ti*4; 
            // clamping 
            light = light < -10 ? -10 : light > 100 ? 100 : light; 
            data[i4] = texture.data[ti4]+light; 
            data[i4+1] = texture.data[ti4+1]+light; 
            data[i4+2] = texture.data[ti4+2]+light; 
        } 
    } 
    aux = buffer0; 
    buffer0 = buffer1; 
    buffer1 = aux; 
    ctx.putImageData(img, 0, 0); 
} 
var canvas = document.getElementById('c'), 
    ctx = canvas.getContext('2d'), 
    width = 320, 
    height = 240, 
    size = width*height, 
    buffer0 = [], 
    buffer1 = [], 
    aux, i, texture; 
 
for(i=0;i<size;i++){ 
    buffer0.push(0); 
    buffer1.push(0); 
} 
 
loadImage(function(img){ 
    texture = getDataFromImage(img); 
    canvas.width = width; 
    canvas.height = height; 
    ctx.fillRect(0, 0, width, height); 
    setInterval(process, 1000/60); 
    resizeBy(width-innerWidth, height-innerHeight); 
}); 
 
canvas.onmousemove = function(e){ 
    disturb( 
            floor(e.clientX/innerWidth*width), 
            floor(e.clientY/innerHeight*height), 
            15000); 
}
<img id="bg" src="" style="display:none;"> 
<canvas id="c" style="position: absolute; top:0; left:0; width:100%;height:100%; "></canvas>

READ ALSO
Масштабируемый clipPath

Масштабируемый clipPath

Суть в том, что svg фигура в clipPath отрисовывается как есть, по тем самым размерам, с которыми создана

91
использовать другой маркер

использовать другой маркер

Использую либу highchartsВстал вопрос - как вместо маркеров использовать диаграмму

108
Правильный import/export React

Правильный import/export React

Подскажите как правильно експортировать и импортировать несколько const

112