• Jump To … +
    ./demo/canvas-001.js ./demo/canvas-002.js ./demo/canvas-003.js ./demo/canvas-004.js ./demo/canvas-005.js ./demo/canvas-006.js ./demo/canvas-007.js ./demo/canvas-008.js ./demo/canvas-009.js ./demo/canvas-010.js ./demo/canvas-011.js ./demo/canvas-012.js ./demo/canvas-013.js ./demo/canvas-014.js ./demo/canvas-015.js ./demo/canvas-015a.js ./demo/canvas-016.js ./demo/canvas-017.js ./demo/canvas-018.js ./demo/canvas-019.js ./demo/canvas-019a.js ./demo/canvas-020.js ./demo/canvas-021.js ./demo/canvas-022.js ./demo/canvas-023.js ./demo/canvas-024.js ./demo/canvas-025.js ./demo/canvas-026.js ./demo/canvas-027.js ./demo/canvas-028.js ./demo/canvas-029.js ./demo/canvas-030.js ./demo/canvas-031.js ./demo/canvas-032.js ./demo/canvas-033.js ./demo/canvas-034.js ./demo/canvas-035.js ./demo/canvas-036.js ./demo/canvas-037.js ./demo/canvas-038.js ./demo/canvas-039.js ./demo/canvas-040.js ./demo/canvas-041.js ./demo/canvas-042.js ./demo/canvas-043.js ./demo/canvas-044.js ./demo/canvas-045.js ./demo/canvas-046.js ./demo/canvas-047.js ./demo/canvas-048.js ./demo/canvas-049.js ./demo/canvas-050.js ./demo/canvas-051.js ./demo/canvas-052.js ./demo/canvas-053.js ./demo/canvas-054.js ./demo/canvas-055.js ./demo/canvas-056.js ./demo/canvas-057.js ./demo/canvas-058.js ./demo/canvas-059.js ./demo/canvas-060.js ./demo/canvas-061.js ./demo/canvas-062.js ./demo/canvas-063.js ./demo/canvas-064.js ./demo/core-001.js ./demo/delaunator-001.js ./demo/delaunator-002.js ./demo/dom-001.js ./demo/dom-002.js ./demo/dom-003.js ./demo/dom-004.js ./demo/dom-005.js ./demo/dom-006.js ./demo/dom-007.js ./demo/dom-008.js ./demo/dom-009.js ./demo/dom-010.js ./demo/dom-011.js ./demo/dom-012.js ./demo/dom-013.js ./demo/dom-015.js ./demo/dom-016.js ./demo/filters-001.js ./demo/filters-002.js ./demo/filters-002a.js ./demo/filters-003.js ./demo/filters-004.js ./demo/filters-005.js ./demo/filters-006.js ./demo/filters-007.js ./demo/filters-008.js ./demo/filters-009.js ./demo/filters-010.js ./demo/filters-011.js ./demo/filters-012.js ./demo/filters-013.js ./demo/filters-014.js ./demo/filters-015.js ./demo/filters-016.js ./demo/filters-017.js ./demo/filters-018.js ./demo/filters-019.js ./demo/filters-020.js ./demo/filters-021.js ./demo/filters-022.js ./demo/filters-023.js ./demo/filters-024.js ./demo/filters-025.js ./demo/filters-026.js ./demo/filters-027.js ./demo/filters-101.js ./demo/filters-102.js ./demo/filters-103.js ./demo/filters-104.js ./demo/filters-105.js ./demo/filters-501.js ./demo/filters-502.js ./demo/filters-503.js ./demo/filters-504.js ./demo/filters-505.js ./demo/mediapipe-001.js ./demo/mediapipe-002.js ./demo/mediapipe-003.js ./demo/modules-001.js ./demo/modules-002.js ./demo/modules-003.js ./demo/modules-004.js ./demo/modules-005.js ./demo/packets-001.js ./demo/packets-002.js ./demo/particles-001.js ./demo/particles-002.js ./demo/particles-003.js ./demo/particles-004.js ./demo/particles-005.js ./demo/particles-006.js ./demo/particles-007.js ./demo/particles-008.js ./demo/particles-009.js ./demo/particles-010.js ./demo/particles-011.js ./demo/particles-012.js ./demo/particles-013.js ./demo/particles-014.js ./demo/particles-015.js ./demo/particles-016.js ./demo/rapier-001.js ./demo/snippets-001.js ./demo/snippets-002.js ./demo/snippets-003.js ./demo/snippets-004.js ./demo/snippets-005.js ./demo/snippets-006.js ./demo/temp-000.js ./demo/temp-001.js ./demo/temp-001a.js ./demo/temp-002.js ./demo/temp-003.js ./demo/temp-004.js ./demo/temp-005.js ./demo/temp-006.js ./demo/temp-007.js ./demo/temp-008.js ./demo/temp-009.js ./demo/temp-010.js ./demo/temp-011.js ./demo/temp-012.js ./demo/temp-013.js ./demo/temp-014.js ./demo/temp-015.js ./demo/temp-016.js ./demo/temp-017.js ./demo/temp-018.js ./demo/temp-018a.js ./demo/temp-019.js ./demo/temp-020.js ./demo/temp-021.js ./demo/temp-022.js ./demo/temp-023.js ./demo/temp-024.js ./demo/temp-025.js ./demo/temp-026.js ./demo/temp-027.js ./demo/temp-028.js ./demo/temp-029.js ./demo/temp-030.js ./demo/temp-031.js ./demo/temp-032.js ./demo/temp-033.js ./demo/temp-034.js ./demo/temp-035.js ./demo/temp-036.js ./demo/temp-037.js ./demo/temp-100.js ./demo/temp-101.js ./demo/temp-102.js ./demo/temp-103.js ./demo/temp-104.js ./demo/temp-200.js ./demo/temp-201.js ./demo/temp-301.js ./demo/temp-inkscapeSvgFilters.js ./demo/temp-lottie.js ./demo/tensorflow-001.js ./demo/tensorflow-002.js ./demo/utilities.js
  • §

    Demo Filters 104

    Compound filters: Use a white-transparent gradient as a filter input

  • §

    Run code

    import * as scrawl from '../source/scrawl.js';
    
    import { reportSpeed, addImageDragAndDrop } from './utilities.js';
  • §

    Scene setup

    const canvas = scrawl.library.canvas.mycanvas;
  • §

    The base cell needs to compile after the helper cells we’re about to create

    canvas.setBase({
        compileOrder: 1,
    });
  • §

    Gradients and pattern definitions

    We display the gradients in a Block entity, in a dedicated Cell

    • The Cell will act as a pattern source for entitys elsewhere
    • Because we’re using it as a pattern, the Cell comes with a pattern matrix
    • The pattern matrix lets us warp and stretch the pattern to meet our needs
    const patternCell = canvas.buildCell({
    
        name: 'bar-cell-pattern',
        dimensions: [80, 80],
        shown: false,
        useAsPattern: true,
        skewX: 0,
        stretchX: 1,
        shiftX: 0,
        skewY: 0,
        stretchY: 1,
        shiftY: 0,
    });
    
    const barGradient = scrawl.makeGradient({
    
        name: 'bar-cell-gradient',
        endX: '100%',
    
        colors: [
            [0, 'transparent'],
            [199, 'transparent'],
            [499, 'white'],
            [799, 'transparent'],
            [999, 'transparent'],
        ],
    });
    
    const ringGradient = scrawl.makeRadialGradient({
    
        name: 'ring-cell-gradient',
        startX: '50%',
        startY: '50%',
        endX: '50%',
        endY: '50%',
    
        startRadius: '0%',
        endRadius: '50%',
    
        colors: [
            [0, 'transparent'],
            [199, 'transparent'],
            [499, 'white'],
            [799, 'transparent'],
            [999, 'transparent'],
        ],
    });
    
    const updateGradients = (items) => {
    
        barGradient.set(items);
        ringGradient.set(items);
    };
    
    const filterBlock = scrawl.makeBlock({
    
        name: 'bar-cell-box',
        group: 'bar-cell-pattern',
        dimensions: ['100%', '100%'],
        fillStyle: 'bar-cell-gradient',
    });
  • §

    Build an image source for our compound filter

    We can use anything that generates image data as part of our filter

    • This includes a new Cell, which will display a Block entity
    • The Block entity uses our pattern Cell for its fillStyle attribute
    const filterCell = canvas.buildCell({
    
        name: 'bar-cell',
        dimensions: ['100%', '100%'],
        shown: false,
    });
    
    scrawl.makeBlock({
    
        name: 'bar-base-box',
        group: 'bar-cell',
        dimensions: ['100%', '100%'],
        fillStyle: 'bar-cell-pattern',
    });
  • §

    Filter definitions

    Define our filters - we’ll populate them with actions data later

    const glassBarsFilter = scrawl.makeFilter({ name: 'glass-bars' }),
        displaceBarsFilter = scrawl.makeFilter({ name: 'displace-bars' }),
        etchingFilter = scrawl.makeFilter({ name: 'etching' }),
        greenMonitorFilter = scrawl.makeFilter({ name: 'green-monitor' });
  • §

    We define some variables and helper functions here, to cut down on code

    let fOffsetX = 0,
        fOffsetY = 0,
        fOpacity = 1,
        fBlend = 'multiply',
        fDisplaceX = 10,
        fDisplaceY = 10;
  • §

    All the examples make use of the gradient pattern; this filter loads it into the engine

    const getProcessImageFilter = () => {
        return [{
            action: 'process-image',
            asset: 'bar-cell',
            width: '100%',
            height: '100%',
            copyWidth: '100%',
            copyHeight: '100%',
            lineOut: 'bars',
        }];
    };
  • §

    Three of the examples use a blend filter to merge the gradient pattern into the image

    const getBlendFilter = () => {
        return [{
            action: 'blend',
            lineMix: 'bars',
            blend: fBlend,
            offsetX: fOffsetX,
            offsetY: fOffsetY,
            opacity: fOpacity,
        }];
    };
  • §

    This uses the gradient pattern as a displacement map for the displace filter

    const getDisplaceFilter = () => {
        return [{
            action: 'displace',
            lineMix: 'bars',
            offsetX: fOffsetX,
            offsetY: fOffsetY,
            opacity: fOpacity,
            scaleX: fDisplaceX,
            scaleY: fDisplaceY,
        }];
    };
  • §

    Several filters require the image to be grayscaled at the start of the processing chain

    const getGrayscaleFilter = () => {
        return [{
            action: 'grayscale',
        }];
    };
  • §

    Filters to configure the image (make it green) before applying the blend filter

    const getMonitorFilter = () => {
        return [{
            action: 'modulate-channels',
            red: 0,
            green: 1.3,
            blue: 0,
        }];
    };
  • §

    Filters to configure the image (give it a black-white drawing effect) before applying the blend filter

    const getEtchFilter = () => {
        return [{
            action: 'gaussian-blur',
            radius: 1,
        }, {
            action: 'matrix',
            weights: [1, 1, 1, 1, -8, 1, 1, 1, 1],
        }, {
            action: 'invert-channels',
        }, {
            action: 'threshold',
            level: 252,
        }];
    };
  • §

    Helper function to update filter action arrays with user choices

    const updateFilters = () => {
    
        glassBarsFilter.set({
            actions: [
                ...getProcessImageFilter(),
                ...getBlendFilter(),
            ],
        });
    
        displaceBarsFilter.set({
            actions: [
                ...getProcessImageFilter(),
                ...getDisplaceFilter(),
            ],
        });
    
        etchingFilter.set({
            actions: [
                ...getProcessImageFilter(),
                ...getGrayscaleFilter(),
                ...getEtchFilter(),
                ...getBlendFilter(),
            ],
        });
    
        greenMonitorFilter.set({
            actions: [
                ...getProcessImageFilter(),
                ...getGrayscaleFilter(),
                ...getMonitorFilter(),
                ...getBlendFilter(),
            ],
        });
    };
  • §

    Initial filter build

    updateFilters();
  • §

    Build the scene

    The initial image

    scrawl.importDomImage('.flowers');
  • §

    We need something to apply our filters to

    const target = scrawl.makePicture({
    
        name: 'target-image',
    
        asset: 'iris',
    
        width: '100%',
        height: '100%',
    
        copyWidth: '100%',
        copyHeight: '100%',
    
        method: 'fill',
    
        filters: ['glass-bars'],
    });
  • §

    Add some Drag-and-Drop image loading functionality

    • So users can check the filter effects against different images
    addImageDragAndDrop(canvas, '#my-image-store', target);
  • §

    Scene animation

    Function to display frames-per-second data, and other information relevant to the demo

    const report = reportSpeed('#reportmessage', function () {
  • §

    @ts-expect-error

        return `    Filter dimensions - \n        width: ${filterWidth.value}%; height: ${filterHeight.value}%\n    Filter offset - \n        x: ${filterOffsetX.value}px; y: ${filterOffsetY.value}px\n    Pattern dimensions - \n        width: ${patternWidth.value}px; height: ${patternHeight.value}px\n    Pattern matrix -\n        shiftX: ${shiftX.value}; shiftY: ${shiftY.value}\n        skewX: ${skewX.value}; skewY: ${skewY.value}\n        stretchX: ${stretchX.value}; stretchY: ${stretchY.value}\n    Displace filter scaling - \n        x: ${filterScaleX.value}; y: ${filterScaleY.value}\n    Opacity: ${opacity.value}`;
    });
  • §

    Create the Display cycle animation

    const demoAnimation = scrawl.makeRender({
    
        name: "demo-animation",
        target: canvas,
        afterShow: report,
    });
  • §

    User interaction

    Update pattern values

    scrawl.observeAndUpdate({
    
        event: ['change', 'input'],
        origin: '.patternCell',
    
        target: patternCell,
    
        useNativeListener: true,
        preventDefault: true,
    
        updates: {
            'pattern-width': ['width', 'round'],
            'pattern-height': ['height', 'round'],
            shiftX: ['shiftX', 'round'],
            shiftY: ['shiftY', 'round'],
            skewX: ['skewX', 'float'],
            skewY: ['skewY', 'float'],
            stretchX: ['stretchX', 'float'],
            stretchY: ['stretchY', 'float'],
        },
    });
  • §

    Update filter dimensions

    scrawl.observeAndUpdate({
    
        event: ['change', 'input'],
        origin: '.filterCell',
    
        target: filterCell,
    
        useNativeListener: true,
        preventDefault: true,
    
        updates: {
            'filter-width': ['width', '%'],
            'filter-height': ['height', '%'],
        },
    });
  • §

    Update gradient choice

    scrawl.observeAndUpdate({
    
        event: ['change', 'input'],
        origin: '.filterBlock',
    
        target: filterBlock,
    
        useNativeListener: true,
        preventDefault: true,
    
        updates: {
            patternGradient: ['fillStyle', 'raw'],
        },
    });
  • §

    Update filter offset, opacity, blend choices

    scrawl.addNativeListener(['change', 'input'], (e) => {
    
        const t = e.target,
            value = t.value;
    
        switch (t.id) {
    
            case 'filter-offset-x' :
                fOffsetX = parseInt(value, 10);
                break;
    
            case 'filter-offset-y' :
                fOffsetY = parseInt(value, 10);
                break;
    
            case 'filter-scale-x' :
                fDisplaceX = parseInt(value, 10);
                break;
    
            case 'filter-scale-y' :
                fDisplaceY = parseInt(value, 10);
                break;
    
            case 'opacity' :
                fOpacity = parseFloat(value);
                break;
    
            case 'blend' :
                fBlend = value;
                break;
        }
        updateFilters();
    
    }, '.filterObject');
  • §

    Update gradient easing

    scrawl.addNativeListener(['change', 'input'], (e) => {
    
        const t = e.target,
            value = t.value;
    
        updateGradients({
            easing: value,
        });
    
    }, '.gradient');
  • §

    Changing the filter effect

    We do quite a bit of work here to set things up for the example’s initial display

    • This includes setting the user controls to match the initial settings
    scrawl.addNativeListener(['change', 'input'], (e) => {
    
        const t = e.target,
            value = t.value;
    
        switch (value) {
    
            case 'glass-bars' :
                fOffsetX = 0;
                fOffsetY = 0;
                fOpacity = 1;
                fBlend = 'multiply';
                fDisplaceX = 10;
                fDisplaceY = 10;
    
                patternCell.set({
                    width: 80,
                    height: 80,
                    shiftX: 0,
                    shiftY: 0,
                    skewX: 0,
                    skewY: 0,
                    stretchX: 1,
                    stretchY: 1,
                });
    
                filterBlock.set({
                    fillStyle: 'bar-cell-gradient',
                });
  • §

    @ts-expect-error

                skewX.value = 0;
  • §

    @ts-expect-error

                skewY.value = 0;
  • §

    @ts-expect-error

                stretchX.value = 1;
  • §

    @ts-expect-error

                stretchY.value = 1;
  • §

    @ts-expect-error

                shiftX.value = 0;
  • §

    @ts-expect-error

                shiftY.value = 0;
  • §

    @ts-expect-error

                filterBlend.value = 'multiply';
  • §

    @ts-expect-error

                patternGradient.value = 'bar-cell-gradient';
  • §

    @ts-expect-error

                patternWidth.value = 80;
  • §

    @ts-expect-error

                patternHeight.value = 80;
    
                break;
    
            case 'displace-bars' :
                fOffsetX = 0;
                fOffsetY = 0;
                fOpacity = 1;
                fBlend = 'multiply';
                fDisplaceX = 10;
                fDisplaceY = 10;
    
                patternCell.set({
                    width: 80,
                    height: 80,
                    shiftX: 0,
                    shiftY: 0,
                    skewX: 0,
                    skewY: 0,
                    stretchX: 1,
                    stretchY: 1,
                });
    
                filterBlock.set({
                    fillStyle: 'ring-cell-gradient',
                });
  • §

    @ts-expect-error

                skewX.value = 0;
  • §

    @ts-expect-error

                skewY.value = 0;
  • §

    @ts-expect-error

                stretchX.value = 1;
  • §

    @ts-expect-error

                stretchY.value = 1;
  • §

    @ts-expect-error

                shiftX.value = 0;
  • §

    @ts-expect-error

                shiftY.value = 0;
  • §

    @ts-expect-error

                filterBlend.value = 'multiply';
  • §

    @ts-expect-error

                patternGradient.value = 'ring-cell-gradient';
  • §

    @ts-expect-error

                patternWidth.value = 80;
  • §

    @ts-expect-error

                patternHeight.value = 80;
    
                break;
    
            case 'etching' :
                fOffsetX = 0;
                fOffsetY = 0;
                fOpacity = 1;
                fBlend = 'screen';
                fDisplaceX = 10;
                fDisplaceY = 10;
    
                patternCell.set({
                    width: 80,
                    height: 80,
                    shiftX: 0,
                    shiftY: 0,
                    skewX: -0.65,
                    skewY: -0.31,
                    stretchX: 1,
                    stretchY: 0.26,
                });
    
                filterBlock.set({
                    fillStyle: 'bar-cell-gradient',
                });
  • §

    @ts-expect-error

                skewX.value = -0.65;
  • §

    @ts-expect-error

                skewY.value = -0.31;
  • §

    @ts-expect-error

                stretchX.value = 1;
  • §

    @ts-expect-error

                stretchY.value = 0.26;
  • §

    @ts-expect-error

                shiftX.value = 0;
  • §

    @ts-expect-error

                shiftY.value = 0;
  • §

    @ts-expect-error

                filterBlend.value = 'screen';
  • §

    @ts-expect-error

                patternGradient.value = 'bar-cell-gradient';
  • §

    @ts-expect-error

                patternWidth.value = 80;
  • §

    @ts-expect-error

                patternHeight.value = 80;
    
                break;
    
            case 'green-monitor' :
                fOffsetX = 0;
                fOffsetY = 0;
                fOpacity = 1;
                fBlend = 'luminosity';
                fDisplaceX = 10;
                fDisplaceY = 10;
    
                patternCell.set({
                    width: 80,
                    height: 80,
                    shiftX: 0,
                    shiftY: 0,
                    skewX: 1.04,
                    skewY: 0.05,
                    stretchX: 0.35,
                    stretchY: 0,
                });
    
                filterBlock.set({
                    fillStyle: 'bar-cell-gradient',
                });
  • §

    @ts-expect-error

                skewX.value = 1.04;
  • §

    @ts-expect-error

                skewY.value = 0.05;
  • §

    @ts-expect-error

                stretchX.value = 0.35;
  • §

    @ts-expect-error

                stretchY.value = 0;
  • §

    @ts-expect-error

                shiftX.value = 0;
  • §

    @ts-expect-error

                shiftY.value = 0;
  • §

    @ts-expect-error

                filterBlend.value = 'luminosity';
  • §

    @ts-expect-error

                patternGradient.value = 'bar-cell-gradient';
  • §

    @ts-expect-error

                patternWidth.value = 80;
  • §

    @ts-expect-error

                patternHeight.value = 80;
    
                break;
        }
    
        filterCell.set({
            width: '100%',
            height: '100%',
        });
    
        updateFilters();
    
        updateGradients({
            easing: 'linear',
        });
  • §

    @ts-expect-error

        opacity.value = 1;
  • §

    @ts-expect-error

        filterWidth.value = 100;
  • §

    @ts-expect-error

        filterHeight.value = 100;
  • §

    @ts-expect-error

        filterOffsetX.value = 0;
  • §

    @ts-expect-error

        filterOffsetY.value = 0;
  • §

    @ts-expect-error

        filterScaleX.value = 10;
  • §

    @ts-expect-error

        filterScaleY.value = 10;
  • §

    @ts-expect-error

        easing.value = 'linear';
  • §

    @ts-expect-error

        target.clearFilters().addFilters(value);
    
    }, '.filterEffect');
  • §

    Setup form on page load

    const opacity = document.querySelector('#opacity');
    const skewX = document.querySelector('#skewX');
    const skewY = document.querySelector('#skewY');
    const stretchX = document.querySelector('#stretchX');
    const stretchY = document.querySelector('#stretchY');
    const shiftX = document.querySelector('#shiftX');
    const shiftY = document.querySelector('#shiftY');
    const patternWidth = document.querySelector('#pattern-width');
    const patternHeight = document.querySelector('#pattern-height');
    const filterWidth = document.querySelector('#filter-width');
    const filterHeight = document.querySelector('#filter-height');
    const filterOffsetX = document.querySelector('#filter-offset-x');
    const filterOffsetY = document.querySelector('#filter-offset-y');
    const filterScaleX = document.querySelector('#filter-scale-x');
    const filterScaleY = document.querySelector('#filter-scale-y');
    const filterBlend = document.querySelector('#blend');
    const patternGradient = document.querySelector('#patternGradient');
    const easing = document.querySelector('#easing');
  • §

    @ts-expect-error

    opacity.value = 1;
  • §

    @ts-expect-error

    skewX.value = 0;
  • §

    @ts-expect-error

    skewY.value = 0;
  • §

    @ts-expect-error

    stretchX.value = 1;
  • §

    @ts-expect-error

    stretchY.value = 1;
  • §

    @ts-expect-error

    shiftX.value = 0;
  • §

    @ts-expect-error

    shiftY.value = 0;
  • §

    @ts-expect-error

    patternWidth.value = 80;
  • §

    @ts-expect-error

    patternHeight.value = 80;
  • §

    @ts-expect-error

    filterWidth.value = 100;
  • §

    @ts-expect-error

    filterHeight.value = 100;
  • §

    @ts-expect-error

    filterOffsetX.value = 0;
  • §

    @ts-expect-error

    filterOffsetY.value = 0;
  • §

    @ts-expect-error

    filterScaleX.value = 10;
  • §

    @ts-expect-error

    filterScaleY.value = 10;
  • §

    @ts-expect-error

    filterBlend.value = 'multiply';
  • §

    @ts-expect-error

    patternGradient.value = 'bar-cell-gradient';
  • §

    @ts-expect-error

    easing.value = 'linear';
  • §

    @ts-expect-error

    document.querySelector('#filterEffect').value = 'glass-bars';
  • §

    Development and testing

    console.log(scrawl.library);